/*****************************************************************************
*                                                                            *
*  Vertex-invariants source file for nauty 2.2.                              *
*                                                                            *
*   Copyright (1989-2002) Brendan McKay.  All rights reserved.               *
*   Subject to waivers and disclaimers in nauty.h.                           *
*                                                                            *
*   CHANGE HISTORY                                                           *
*       13-Mar-90 : initial release for version 1.5                          *
*       10-Nov-90 : changes for version 1.6 :                                *
*                 - added dummy routine nautinv_null()                       *
*       27-Aug-92 : renamed to version 1.7, no changes to this file          *
*        5-Jun-93 : renamed to version 1.7+, no changes to this file         *
*       18-Aug-93 : renamed to version 1.8, no changes to this file          *
*       17-Sep-93 : changes for version 1.9 :                                *
*                 - added invariant routine adjacencies()                    *
*       20-Jun-96 : changes for version 2.0 :                                *
*                 - added invariants cellfano() and cellfano2()              *
*       11-Jul-96 - added dynamic allocation                                 *
*       21-Oct-98 - use shortish in place of short for BIGNAUTY              *
*        9-Jan-00 - added nautinv_check()                                    *
*       12-Feb-00 - minor code formatting                                    *
*       16-Nov-00 - made changes listed in nauty.h                           *
*       22-Apr-01 : changes for version 2.1 :                                *
*                 - made all large dynamic memory external to routines       *
*                 - added nautinv_freedyn() to free all such memory          *
*                 - include nautinv.h rather than naututil.h                 *
*                 - removed nautinv_null()                                   *
*                 - added code to facilitate compilation into Magma          *
*                 - removed EXTDEFS                                          *
*       12-Jul-01 - use invararg in distances()                              *
*                 - fixed comments in ind and cliq routines                  *
*       21-Nov-01 : use NAUTYREQUIRED in nautinv_check()                     *
*                                                                            *
*****************************************************************************/

#define ONE_WORD_SETS
#include "nautinv.h"

#if  MAXM==1
#define M 1
#else
#define M m
#endif

#define ACCUM(x,y)   x = (((x) + (y)) & 077777)

static int fuzz1[] = {037541,061532,005257,026416};
static int fuzz2[] = {006532,070236,035523,062437};

#define FUZZ1(x) ((x) ^ fuzz1[(x)&3])
#define FUZZ2(x) ((x) ^ fuzz2[(x)&3])

#define MAXCLIQUE 10    /* max clique size for cliques() and maxindset() */

#if MAXN
static set workset[MAXM];
static shortish workshort[MAXN+2];
static set ws1[MAXM], ws2[MAXM];
static shortish vv[MAXN],ww[MAXN];
static set w01[MAXM],w02[MAXM],w03[MAXM],w12[MAXM],w13[MAXM],w23[MAXM];
static set pt0[MAXM],pt1[MAXM],pt2[MAXM];
static set wss[MAXCLIQUE-1][MAXM];
#else
DYNALLSTAT(set,workset,workset_sz);
DYNALLSTAT(shortish,workshort,workshort_sz);
DYNALLSTAT(set,ws1,ws1_sz);
DYNALLSTAT(set,ws2,ws2_sz);
DYNALLSTAT(shortish,vv,vv_sz);
DYNALLSTAT(shortish,ww,ww_sz);
DYNALLSTAT(set,w01,w01_sz);
DYNALLSTAT(set,w02,w02_sz);
DYNALLSTAT(set,w03,w03_sz);
DYNALLSTAT(set,w12,w12_sz);
DYNALLSTAT(set,w13,w13_sz);
DYNALLSTAT(set,w23,w23_sz);
DYNALLSTAT(set,pt0,pt0_sz);
DYNALLSTAT(set,pt1,pt1_sz);
DYNALLSTAT(set,pt2,pt2_sz);
DYNALLSTAT(set,wss,wss_sz);
#endif

/* aproto: header new_nauty_protos.h */

/*****************************************************************************
*                                                                            *
*  This file contains a number of procedures which compute vertex-invariants *
*  for stronger partition refinement.   Since entirely different             *
*  vertex-invariants seem to work better for different types of graph, we    *
*  cannot do more than give a small collection of representative examples.   *
*  Any serious computations with difficult graphs may well need to use       *
*  specially-written vertex-invariants.  The use of vertex-invariants        *
*  procedures is supported by nauty from version 1.5 onwards, via the        *
*  options userinvarproc, mininvarlevel, maxinvarlevel and invararg.         *
*  The meaning of these fields in detail are as follows:                     *
*     userinvarproc  is the address of the vertex-invariant procedure.  If   *
*                    no vertex-invariants is required, this field should     *
*                    have the value NULL.                                    *
*     maxinvarlevel  The absolute value of this is the maximum level in the  *
*                    search tree at which the vertex-invariant will be       *
*                    computed.  The root of the tree is at level 1, so the   *
*                    vertex-invariant will not be invoked at all if          *
*                    maxinvarlevel==0.  Negative values of maxinvarlevel     *
*                    request nauty to not compute the vertex-invariant at    *
*                    a level greater than that of the earliest node (if any) *
*                    on the path to the first leaf of the search tree at     *
*                    which the vertex-invariant refines the partition.       *
*     mininvarlevel  The absolute value of this is the minimum level in the  *
*                    search tree at which the vertex-invariant will be       *
*                    computed.  The root of the tree is at level 1, so there *
*                    is no effective limit if mininvarlevel is -1, 0 or 1.   *
*                    Negative values of mininvarlevel request nauty to not   *
*                    compute the vertex-invariant at a level less than       *
*                    that of the earliest node (if any) on the path to the   *
*                    first leaf of the search tree at which the              *
*                    vertex-invariant refines the partition.                 *
*     invararg       is passed to the vertex-invariant procedure via the     *
*                    argument of the same name.  It can be used by the       *
*                    procedure for any purpose.                              *
*  Note that negative values of maxinvarlevel and mininvarlevel make the     *
*  canonical labelling invalid, but can speed automorphism group finding.    *
*  Nauty already knows this and takes their absolute values.                 *
*                                                                            *
*  A vertex-invariant must be declared thus:                                 *
*  void invarproc(g,lab,ptn,level,numcells,tvpos,invar,invararg,digraph,m,n) *
*  All of these arguments must be treated as read-only except for invar.     *
*  g        : the graph, exactly as passed to nauty()                        *
*  lab,ptn  : the current partition nest (see nauty.h for the format)        *
*  level    : the level of this node in the search tree.                     *
*  numcells : the number of cells in the partition at this node.             *
*  tvpos    : the index in (lab,ptn) of one cell in the partition.           *
*             If level <= 1, the cell will be the first fragment of the      *
*             first active cell (as provided by the initial call to nauty),  *
*             or the first cell, if there were no active cells.              *
*             If level > 1, the cell will be the singleton cell which was    *
*             created to make this node of the search tree from its parent.  *
*  invararg : a copy of options.invararg                                     *
*  digraph  : a copy of options.digraph                                      *
*  m,n      : size parameters as passed to nauty()                           *
*  invar    : an array to return the answer in.   The procedure must put in  *
*             each invar[i]  (0 <= i < n)  an invariant of the 6-tuple       *
*             (<vertex i>,g,<the partition nest to this level>,level,        *
*               invararg,digraph)                                            *
*             Note that invar[] is declared as a permutation.  Since the     *
*             absolute value of the invariant is irrelevant, only the        *
*             comparative values, any short, int or long value can be        *
*             assigned to the entries of invar[] without fear.  However,     *  
*             you should assign a value less than 077777 to ensure machine-  *
*             independence of the canonical labelling.                       *
*                                                                            *
*  The refinement procedure has already been called before the invariant     *
*  procedure is called.  That means that the partition is equitable if       *
*  digraph==FALSE.                                                           *
*                                                                            *
*****************************************************************************/

/*****************************************************************************
*                                                                            *
*  twopaths() assigns to each vertex v the sum of the weights of each vertex *
*  which can be reached from v along a walk of length two (including itself  *
*  usually).  The weight of each vertex w is defined as the ordinal number   *
*  of the cell containing w, starting at 1 for the first cell.               *
*                                                                            *
*****************************************************************************/

void
twopaths(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
         permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i,v,w;
        register shortish wt;
        set *gv,*gw;

#if !MAXN
	DYNALLOC1(set,workset,workset_sz,m,"twopaths");
	DYNALLOC1(shortish,workshort,workshort_sz,n+2,"twopaths");
#endif

        wt = 1;
        for (i = 0; i < n; ++i)
        {
            workshort[lab[i]] = wt;
            if (ptn[i] <= level) ++wt;
        }

        for (v = 0, gv = (set*)g; v < n; ++v, gv += M)
        {
            EMPTYSET(workset,m);
            w = -1;
            while ((w = nextelement(gv,M,w)) >= 0)
            {
                gw = GRAPHROW(g,w,m);
                for (i = M; --i >= 0;) UNION(workset[i],gw[i]);
            }
            wt = 0;
            w = -1;
            while ((w = nextelement(workset,M,w)) >= 0) ACCUM(wt,workshort[w]);
            invar[v] = wt;
        }
}

/*****************************************************************************
*                                                                            *
*  quadruples() assigns to each vertex v a value depending on the set of     *
*  weights w(v,v1,v2,v3), where w(v,v1,v2,v3) depends on the number of       *
*  vertices adjacent to an odd number of {v,v1,v2,v3}, and to the cells      *
*  that v,v1,v2,v3 belong to.  {v,v1,v2,v3} are permitted to range over all  *
*  distinct 4-tuples which contain at least one member in the cell tvpos.    *
*                                                                            *
*****************************************************************************/

void
quadruples(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
           permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i,pc;
        register setword sw;
        register set *gw;
        register shortish wt;
        int v,iv,v1,v2,v3;
        set *gv;
        long wv,wv1,wv2,wv3;

#if !MAXN
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"quadruples");
	DYNALLOC1(set,ws1,ws1_sz,m,"quadruples");
	DYNALLOC1(set,workset,workset_sz,m,"quadruples");
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        wt = 1;
        for (i = 0; i < n; ++i)
        {
            workshort[lab[i]] = FUZZ2(wt);
            if (ptn[i] <= level) ++wt;
        }

        iv = tvpos - 1;
        do
        {
             v = lab[++iv];
             gv = GRAPHROW(g,v,m);
             wv = workshort[v];
             for (v1 = 0; v1 < n-2; ++v1)
             {
                wv1 = workshort[v1];
                if (wv1 == wv && v1 <= v) continue;
                wv1 += wv;
                gw = GRAPHROW(g,v1,m);
                for (i = M; --i >= 0;) workset[i] = gv[i] ^ gw[i];
                for (v2 = v1+1; v2 < n-1; ++v2)
                {
                    wv2 = workshort[v2];
                    if (wv2 == wv && v2 <= v) continue;
                    wv2 += wv1;
                    gw = GRAPHROW(g,v2,m);
                    for (i = M; --i >= 0;) ws1[i] = workset[i] ^ gw[i];
                    for (v3 = v2+1; v3 < n; ++v3)
                    {
                        wv3 = workshort[v3];
                        if (wv3 == wv && v3 <= v) continue;
                        wv3 += wv2;
                        gw = GRAPHROW(g,v3,m);
                        pc = 0;
                        for (i = M; --i >= 0;)
                            if ((sw = ws1[i] ^ gw[i]) != 0) pc += POPCOUNT(sw);
                        wt = (FUZZ1(pc)+wv3) & 077777;
                        wt = FUZZ2(wt);
                        ACCUM(invar[v],wt);
                        ACCUM(invar[v1],wt);
                        ACCUM(invar[v2],wt);
                        ACCUM(invar[v3],wt);
                    }
                }
            }
        }
        while (ptn[iv] > level);
}

/*****************************************************************************
*                                                                            *
*  triples() assigns to each vertex v a value depending on the set of        *
*  weights w(v,v1,v2), where w(v,v1,v2) depends on the number of vertices    *
*  adjacent to an odd number of {v,v1,v2}, and to the cells that             *
*  v,v1,v2 belong to.  {v,v1,v2} are permitted to range over all distinct    *
*  triples which contain at least one member in the cell tvpos.              *
*                                                                            *
*****************************************************************************/

void
triples(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
        permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i,pc;
        register setword sw;
        register set *gw;
        register shortish wt;
        int v,iv,v1,v2;
        set *gv;
        long wv,wv1,wv2;

#if !MAXN
        DYNALLOC1(set,workset,workset_sz,m,"triples");
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"triples");
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        wt = 1;
        for (i = 0; i < n; ++i)
        {
            workshort[lab[i]] = FUZZ1(wt);
            if (ptn[i] <= level) ++wt;
        }

        iv = tvpos - 1;
        do
        {
             v = lab[++iv];
             gv = GRAPHROW(g,v,m);
             wv = workshort[v];
             for (v1 = 0; v1 < n-1; ++v1)
             {
                wv1 = workshort[v1];
                if (wv1 == wv && v1 <= v) continue;
                wv1 += wv;
                gw = GRAPHROW(g,v1,m);
                for (i = M; --i >= 0;) workset[i] = gv[i] ^ gw[i];
                for (v2 = v1+1; v2 < n; ++v2)
                {
                    wv2 = workshort[v2];
                    if (wv2 == wv && v2 <= v) continue;
                    wv2 += wv1;
                    gw = GRAPHROW(g,v2,m);
                    pc = 0;
                    for (i = M; --i >= 0;)
                        if ((sw = workset[i] ^ gw[i]) != 0) pc += POPCOUNT(sw);
                    wt = (FUZZ1(pc)+wv2) & 077777;
                    wt = FUZZ2(wt);
                    ACCUM(invar[v],wt);
                    ACCUM(invar[v1],wt);
                    ACCUM(invar[v2],wt);
                }
            }
        }
        while (ptn[iv] > level);
}

/*****************************************************************************
*                                                                            *
*  adjtriang() assigns to each vertex v a value depending on the numbers     *
*  of common neighbours between each pair {v1,v2} of neighbours of v, and    *
*  which cells v1 and v2 lie in.  The vertices v1 and v2 must be adjacent    *
*  if invararg == 0 and not adjacent if invararg == 1.                       *
*                                                                            *
*****************************************************************************/

void
adjtriang(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
          permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int j,pc;
        register setword sw;
        register set *gi;
        register shortish wt;
        int i,v1,v2;
        boolean v1v2;
        set *gv1,*gv2;

#if !MAXN
        DYNALLOC1(set,workset,workset_sz,m,"adjtriang");
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"adjtriang");
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        wt = 1;
        for (i = 0; i < n; ++i)
        {
            workshort[lab[i]] = FUZZ1(wt);
            if (ptn[i] <= level) ++wt;
        }

        for (v1 = 0, gv1 = g; v1 < n; ++v1, gv1 += M)
        {
            for (v2 = (digraph ? 0 : v1+1); v2 < n; ++v2)
            {
                if (v2 == v1) continue;
                v1v2 = (ISELEMENT(gv1,v2) != 0);
                if (invararg == 0 && !v1v2 || invararg == 1 && v1v2) continue;
                wt = workshort[v1];
                ACCUM(wt,workshort[v2]);
                ACCUM(wt,v1v2);

                gv2 = GRAPHROW(g,v2,m);
                for (i = M; --i >= 0;) workset[i] = gv1[i] & gv2[i];
                i = -1;
                while ((i = nextelement(workset,M,i)) >= 0)
                {
                    pc = 0;
                    gi = GRAPHROW(g,i,m);
                    for (j = M; --j >= 0;)
                        if ((sw = workset[j] & gi[j]) != 0) pc += POPCOUNT(sw);
                    pc = (pc + wt) & 077777;
                    ACCUM(invar[i],pc);
                }
            }
        }
}

/*****************************************************************************
*                                                                            *
*  getbigcells(ptn,level,minsize,bigcells,cellstart,cellsize,n) is an        *
*  auxiliary procedure to make a list of all the large cells in the current  *
*  partition.  On entry, ptn, level and n have their usual meanings,         *
*  while minsize is the smallest size of an interesting cell.  On return,    *
*  bigcells is the number of cells of size at least minsize, cellstart[0...] *
*  contains their starting positions in ptn, and cellsize[0...] contain      *
*  their sizes.  These two arrays are in increasing order of cell size,      *
*  then position.                                                            *
*                                                                            *
*****************************************************************************/

void
getbigcells(int *ptn, int level, int minsize, int *bigcells,
            shortish *cellstart, shortish *cellsize, int n)
{
        register int cell1,cell2,j;
        register shortish si,st;
        int bc,i,h;

        bc = 0;
        for (cell1 = 0; cell1 < n; cell1 = cell2 + 1)
        {
            for (cell2 = cell1; ptn[cell2] > level; ++cell2) {}

            if (cell2 >= cell1 + 3)
            {
                cellstart[bc] = cell1;
                cellsize[bc] = cell2 - cell1 + 1;
                ++bc;
            }
        }
        *bigcells = bc;

        j = bc / 3;
        h = 1;
        do
            h = 3 * h + 1;
        while (h < j);

        do                      /* shell sort */
        {
            for (i = h; i < bc; ++i)
            {
                st = cellstart[i];
                si = cellsize[i];
                for (j = i; cellsize[j-h] > si ||
                            cellsize[j-h] == si && cellstart[j-h] > st; )
                {
                    cellsize[j] = cellsize[j-h];
                    cellstart[j] = cellstart[j-h];
                    if ((j -= h) < h) break;
                }
                cellsize[j] = si;
                cellstart[j] = st;
            }
            h /= 3;
        }
        while (h > 0);
}

/*****************************************************************************
*                                                                            *
*  celltrips() assigns to each vertex v a value depending on the set of      *
*  weights w(v,v1,v2), where w(v,v1,v2) depends on the number of vertices    *
*  adjacent to an odd number of {v,v1,v2}.  {v,v1,v2} are  constrained to    *
*  belong to the same cell.  We try the cells in increasing order of size,   *
*  and stop as soon as any cell splits.                                      *
*                                                                            *
*****************************************************************************/

void
celltrips(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
          permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i,pc;
        register setword sw;
        register set *gw;
        register shortish wt;
        int v,iv,v1,iv1,v2,iv2;
        int icell,bigcells,cell1,cell2;
        shortish *cellstart,*cellsize;
        set *gv;

#if !MAXN
        DYNALLOC1(set,workset,workset_sz,m,"celltrips");
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"celltrips");
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        cellstart = workshort;
        cellsize = workshort + (n/2);
        getbigcells(ptn,level,3,&bigcells,cellstart,cellsize,n);

        for (icell = 0; icell < bigcells; ++icell)
        {
            cell1 = cellstart[icell];
            cell2 = cell1 + cellsize[icell] - 1;
            for (iv = cell1; iv <= cell2 - 2; ++iv)
            {
                v = lab[iv];
                gv = GRAPHROW(g,v,m);
                for (iv1 = iv + 1; iv1 <= cell2 - 1; ++iv1)
                {
                    v1 = lab[iv1];
                    gw = GRAPHROW(g,v1,m);
                    for (i = M; --i >= 0;) workset[i] = gv[i] ^ gw[i];
                    for (iv2 = iv1 + 1; iv2 <= cell2; ++iv2)
                    {
                        v2 = lab[iv2];
                        gw = GRAPHROW(g,v2,m);
                        pc = 0;
                        for (i = M; --i >= 0;)
                            if ((sw = workset[i] ^ gw[i]) != 0)
                                pc += POPCOUNT(sw);
                        wt = FUZZ1(pc);
                        ACCUM(invar[v],wt);
                        ACCUM(invar[v1],wt);
                        ACCUM(invar[v2],wt);
                    }
                }
            }
            wt = invar[lab[cell1]];
            for (i = cell1 + 1; i <= cell2; ++i)
                if (invar[lab[i]] != wt) return;
        }
}

/*****************************************************************************
*                                                                            *
*  cellquads() assigns to each vertex v a value depending on the set of      *
*  weights w(v,v1,v2,v3), where w(v,v1,v2,v3) depends on the number of       *
*  vertices adjacent to an odd number of {v,v1,v2,v3}.  {v,v1,v2,v3} are     *
*  constrained to belong to the same cell.  We try the cells in increasing   *
*  order of size, and stop as soon as any cell splits.                       *
*                                                                            *
*****************************************************************************/

void
cellquads(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
          permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i,pc;
        register setword sw;
        register set *gw;
        register shortish wt;
        int v,iv,v1,iv1,v2,iv2,v3,iv3;
        int icell,bigcells,cell1,cell2;
        shortish *cellstart,*cellsize;
        set *gv;

#if !MAXN
        DYNALLOC1(set,workset,workset_sz,m,"cellquads");
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"cellquads");
	DYNALLOC1(set,ws1,ws1_sz,m,"cellquads");
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        cellstart = workshort;
        cellsize = workshort + (n/2);
        getbigcells(ptn,level,4,&bigcells,cellstart,cellsize,n);

        for (icell = 0; icell < bigcells; ++icell)
        {
            cell1 = cellstart[icell];
            cell2 = cell1 + cellsize[icell] - 1;
            for (iv = cell1; iv <= cell2 - 3; ++iv)
            {
                v = lab[iv];
                gv = GRAPHROW(g,v,m);
                for (iv1 = iv + 1; iv1 <= cell2 - 2; ++iv1)
                {
                    v1 = lab[iv1];
                    gw = GRAPHROW(g,v1,m);
                    for (i = M; --i >= 0;) workset[i] = gv[i] ^ gw[i];
                    for (iv2 = iv1 + 1; iv2 <= cell2 - 1; ++iv2)
                    {
                        v2 = lab[iv2];
                        gw = GRAPHROW(g,v2,m);
                        for (i = M; --i >= 0;) ws1[i] = workset[i] ^ gw[i];
                        for (iv3 = iv2 + 1; iv3 <= cell2; ++iv3)
                        {
                            v3 = lab[iv3];
                            gw = GRAPHROW(g,v3,m);
                            pc = 0;
                            for (i = M; --i >= 0;)
                                if ((sw = ws1[i] ^ gw[i]) != 0)
                                    pc += POPCOUNT(sw);
                            wt = FUZZ1(pc);
                            ACCUM(invar[v],wt);
                            ACCUM(invar[v1],wt);
                            ACCUM(invar[v2],wt);
                            ACCUM(invar[v3],wt);
                        }
                    }
                }
            }
            wt = invar[lab[cell1]];
            for (i = cell1 + 1; i <= cell2; ++i)
                if (invar[lab[i]] != wt) return;
        }
}

/*****************************************************************************
*                                                                            *
*  cellquins() assigns to each vertex v a value depending on the set of      *
*  weights w(v,v1,v2,v3,v4), where w(v,v1,v2,v3,v4) depends on the number    *
*  of vertices adjacent to an odd number of {v,v1,v2,v3,v4}.                 *
*  {v,v1,v2,v3,v4} are constrained to belong to the same cell.  We try the   *
*  cells in increasing order of size, and stop as soon as any cell splits.   *
*                                                                            *
*****************************************************************************/

void
cellquins(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
          permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i,pc;
        register setword sw;
        register set *gw;
        register shortish wt;
        int v,iv,v1,iv1,v2,iv2,v3,iv3,v4,iv4;
        int icell,bigcells,cell1,cell2;
        shortish *cellstart,*cellsize;
        set *gv;

#if !MAXN
        DYNALLOC1(set,workset,workset_sz,m,"cellquins");
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"cellquins");
	DYNALLOC1(set,ws1,ws1_sz,m,"cellquins");
	DYNALLOC1(set,ws2,ws2_sz,m,"cellquins");
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        cellstart = workshort;
        cellsize = workshort + (n/2);
        getbigcells(ptn,level,5,&bigcells,cellstart,cellsize,n);

        for (icell = 0; icell < bigcells; ++icell)
        {
            cell1 = cellstart[icell];
            cell2 = cell1 + cellsize[icell] - 1;
            for (iv = cell1; iv <= cell2 - 4; ++iv)
            {
                v = lab[iv];
                gv = GRAPHROW(g,v,m);
                for (iv1 = iv + 1; iv1 <= cell2 - 3; ++iv1)
                {
                    v1 = lab[iv1];
                    gw = GRAPHROW(g,v1,m);
                    for (i = M; --i >= 0;) workset[i] = gv[i] ^ gw[i];
                    for (iv2 = iv1 + 1; iv2 <= cell2 - 2; ++iv2)
                    {
                        v2 = lab[iv2];
                        gw = GRAPHROW(g,v2,m);
                        for (i = M; --i >= 0;) ws1[i] = workset[i] ^ gw[i];
                        for (iv3 = iv2 + 1; iv3 <= cell2 - 1; ++iv3)
                        {
                            v3 = lab[iv3];
                            gw = GRAPHROW(g,v3,m);
                            for (i = M; --i >= 0;) ws2[i] = ws1[i] ^ gw[i];
                            for (iv4 = iv3 + 1; iv4 <= cell2; ++iv4)
                            {
                                v4 = lab[iv4];
                                gw = GRAPHROW(g,v4,m);
                                pc = 0;
                                for (i = M; --i >= 0;)
                                    if ((sw = ws2[i] ^ gw[i]) != 0)
                                        pc += POPCOUNT(sw);
                                wt = FUZZ1(pc);
                                ACCUM(invar[v],wt);
                                ACCUM(invar[v1],wt);
                                ACCUM(invar[v2],wt);
                                ACCUM(invar[v3],wt);
                                ACCUM(invar[v4],wt);
                            }
                        }
                    }
                }
            }
            wt = invar[lab[cell1]];
            for (i = cell1 + 1; i <= cell2; ++i)
                if (invar[lab[i]] != wt) return;
        }
}

/*****************************************************************************
*                                                                            *
*  uniqinter(s1,s2,m) returns the number in both sets if it is unique,       *
*  or -1 if there is none or it is not unique.                               *
*****************************************************************************/

static int
uniqinter(set *s1, set *s2, int m)
{
	register int i,j;
	register setword w;

	for (i = 0; i < M; ++i)
	{
	    if ((w = s1[i] & s2[i]) != 0)
	    {
		j = FIRSTBIT(w);
		if (w != BITT[j]) return -1;
		j += TIMESWORDSIZE(i);
		for (++i; i < M; ++i)
		    if (s1[i] & s2[i]) return -1;
		return j;
	    }
	}
	return -1;
}

/*****************************************************************************
*                                                                            *
*  cellfano2() assigns to each vertex v a value depending on the set of      *
*  weights w(v,v1,v2,v3), where w(v,v1,v2,v3) depends on the number of       *
*  fano-plane analogues containing {v,v1,v2,v3}.  {v,v1,v2,v3} are           *
*  constrained to belong to the same cell and being independent and          *
*  non-collinear.  We try the cells in increasing order of size, and stop    *
*  as soon as any cell splits.                                               *
*                                                                            *
*****************************************************************************/

void 
cellfano2(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
          permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i,pc;
        register setword sw;
        register shortish wt;
        int v0,v1,v2,v3,iv0,iv1,iv2,iv3;
        int icell,bigcells,cell1,cell2;
        shortish *cellstart,*cellsize;
	int nw,x01,x02,x03,x12,x13,x23;
	int pnt0,pnt1,pnt2;
	set *gv0,*gv1,*gv2,*gv3;
	set *gp0,*gp1,*gp2;

#if !MAXN
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"cellfano2");
	DYNALLOC1(shortish,vv,vv_sz,n,"cellfano2");
	DYNALLOC1(shortish,ww,ww_sz,n,"cellfano2");
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        cellstart = workshort;
        cellsize = workshort + (n/2);
        getbigcells(ptn,level,4,&bigcells,cellstart,cellsize,n);

        for (icell = 0; icell < bigcells; ++icell)
        {
            cell1 = cellstart[icell];
            cell2 = cell1 + cellsize[icell] - 1;
            for (iv0 = cell1; iv0 <= cell2 - 3; ++iv0)
            {
                v0 = lab[iv0];
                gv0 = GRAPHROW(g,v0,m);
		nw = 0;
		for (iv1 = iv0 + 1; iv1 <= cell2; ++iv1)
		{
		    v1 = lab[iv1];
                    if (ISELEMENT(gv0,v1)) continue;
                    if ((x01 = uniqinter(gv0,GRAPHROW(g,v1,m),m)) < 0) continue;
		    vv[nw] = v1;
		    ww[nw] = x01;
		    ++nw;
		}	

                for (iv1 = 0; iv1 < nw-2; ++iv1)
                {
                    v1 = vv[iv1];
                    gv1 = GRAPHROW(g,v1,m);
		    x01 = ww[iv1];

                    for (iv2 = iv1 + 1; iv2 < nw-1; ++iv2)
                    {
			x02 = ww[iv2];
			if (x02 == x01) continue;
                        v2 = vv[iv2];
			if (ISELEMENT(gv1,v2)) continue;
                        gv2 = GRAPHROW(g,v2,m);
			if ((x12 = uniqinter(gv1,gv2,m)) < 0) continue;

                        for (iv3 = iv2 + 1; iv3 < nw; ++iv3)
                        {
			    x03 = ww[iv3];
			    if (x03 == x01 || x03 == x02) continue;
                            v3 = vv[iv3];
			    if (ISELEMENT(gv1,v3) || ISELEMENT(gv2,v3))
				continue;
                            gv3 = GRAPHROW(g,v3,m);
			    if ((x13 = uniqinter(gv1,gv3,m)) < 0) continue;
			    if ((x23 = uniqinter(gv2,gv3,m)) < 0
                                                   || x23 == x13) continue;

			    if ((pnt0 = uniqinter(GRAPHROW(g,x01,m),
						 GRAPHROW(g,x23,m),m)) < 0)
				continue;
			    if ((pnt1 = uniqinter(GRAPHROW(g,x02,m),
                                                 GRAPHROW(g,x13,m),m)) < 0)
                                continue;
                            if ((pnt2 = uniqinter(GRAPHROW(g,x03,m),
                                                 GRAPHROW(g,x12,m),m)) < 0)
                                continue;

			    gp0 = GRAPHROW(g,pnt0,m);
			    gp1 = GRAPHROW(g,pnt1,m);
			    gp2 = GRAPHROW(g,pnt2,m);

			    pc = 0;
			    for (i = M; --i >= 0;)
			    {
				sw = gp0[i] & gp1[i] & gp2[i];
				if (sw) pc += POPCOUNT(sw);
			    }
			    wt = FUZZ1(pc);
			    ACCUM(invar[v0],wt);
			    ACCUM(invar[v1],wt);
                            ACCUM(invar[v2],wt);
                            ACCUM(invar[v3],wt);
                        }
                    }
                }
            }
            wt = invar[lab[cell1]];
            for (i = cell1 + 1; i <= cell2; ++i)
                if (invar[lab[i]] != wt) return;
        }
}

/*****************************************************************************
*                                                                            *
*  setnbhd(g,m,n,w,wn) is an auxiliary routine that sets wn to the union     *
*  of the neighbours of the vertices in w.                                   *
*                                                                            *
*****************************************************************************/

void
setnbhd(graph *g, int m, int n, set *w, set *wn)
{
	register int i,j;
	register set *gi;

	i = nextelement(w,M,-1);
	if (i < 0)
	{
	    EMPTYSET(wn,M);
	    return;
	}

	gi = GRAPHROW(g,i,M);
	for (j = M; --j >= 0;) wn[j] = gi[j];

	while ((i = nextelement(w,M,i)) >= 0)
	{
	    gi = GRAPHROW(g,i,M);
            for (j = M; --j >= 0;) wn[j] |= gi[j];
	}
}    

/*****************************************************************************
*                                                                            *
*  cellfano() assigns to each vertex v a value depending on the set of       *
*  weights w(v,v1,v2,v3), where w(v,v1,v2,v3) depends on the number of       *
*  fano-plane analogues containing {v,v1,v2,v3}.  {v,v1,v2,v3} are           *
*  constrained to belong to the same cell and being independent.  We try     *
*  the cells in increasing order of size, and stop as soon as any cell       *
*  splits.                                                                   *
*                                                                            *
*****************************************************************************/

void 
cellfano(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
         permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i,pc;
        register setword sw;
        register shortish wt;
        int v0,v1,v2,v3,iv0,iv1,iv2,iv3;
        int icell,bigcells,cell1,cell2;
        shortish *cellstart,*cellsize;
	register set *gv0,*gv1,*gv2,*gv3;

#if !MAXN
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"cellfano");
	DYNALLOC1(set,w01,w01_sz,m,"cellfano");
	DYNALLOC1(set,w02,w02_sz,m,"cellfano");
	DYNALLOC1(set,w03,w03_sz,m,"cellfano");
	DYNALLOC1(set,w12,w12_sz,m,"cellfano");
	DYNALLOC1(set,w13,w13_sz,m,"cellfano");
	DYNALLOC1(set,w23,w23_sz,m,"cellfano");
	DYNALLOC1(set,pt0,pt0_sz,m,"cellfano");
	DYNALLOC1(set,pt1,pt1_sz,m,"cellfano");
	DYNALLOC1(set,pt2,pt2_sz,m,"cellfano");
	DYNALLOC1(set,workset,workset_sz,m,"cellfano");
#else
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        cellstart = workshort;
        cellsize = workshort + (n/2);
        getbigcells(ptn,level,4,&bigcells,cellstart,cellsize,n);

        for (icell = 0; icell < bigcells; ++icell)
        {
            cell1 = cellstart[icell];
            cell2 = cell1 + cellsize[icell] - 1;
            for (iv0 = cell1; iv0 <= cell2 - 3; ++iv0)
            {
                v0 = lab[iv0];
                gv0 = GRAPHROW(g,v0,m);
                for (iv1 = iv0 + 1; iv1 <= cell2 - 2; ++iv1)
                {
                    v1 = lab[iv1];
		    if (ISELEMENT(gv0,v1)) continue;
                    gv1 = GRAPHROW(g,v1,m);
                    for (i = M; --i >= 0;) workset[i] = gv0[i] & gv1[i];
		    setnbhd(g,m,n,workset,w01);

                    for (iv2 = iv1 + 1; iv2 <= cell2 - 1; ++iv2)
                    {
                        v2 = lab[iv2];
			if (ISELEMENT(gv0,v2) || ISELEMENT(gv1,v2))
			    continue;
                        gv2 = GRAPHROW(g,v2,m);
			for (i = M; --i >= 0;) workset[i] = gv0[i] & gv2[i];
                        setnbhd(g,m,n,workset,w02);
                        for (i = M; --i >= 0;) workset[i] = gv1[i] & gv2[i];
                        setnbhd(g,m,n,workset,w12);

                        for (iv3 = iv2 + 1; iv3 <= cell2; ++iv3)
                        {
                            v3 = lab[iv3];
			    if (ISELEMENT(gv0,v3) || ISELEMENT(gv1,v3) ||
					ISELEMENT(gv2,v3))
				continue;
                            gv3 = GRAPHROW(g,v3,m);
                            for (i = M; --i >= 0;) workset[i] = gv0[i] & gv3[i];
                            setnbhd(g,m,n,workset,w03);
                            for (i = M; --i >= 0;) workset[i] = gv1[i] & gv3[i];
                            setnbhd(g,m,n,workset,w13);
                            for (i = M; --i >= 0;) workset[i] = gv2[i] & gv3[i];
                            setnbhd(g,m,n,workset,w23);
			
			    for (i = M; --i >= 0;) workset[i] = w01[i] & w23[i];
			    setnbhd(g,m,n,workset,pt0);
                            for (i = M; --i >= 0;) workset[i] = w03[i] & w12[i];
                            setnbhd(g,m,n,workset,pt1);
                            for (i = M; --i >= 0;) workset[i] = w02[i] & w13[i];
                            setnbhd(g,m,n,workset,pt2);
			    pc = 0;
			    for (i = M; --i >= 0;)
			    {
				sw = pt0[i] & pt1[i] & pt2[i];
				if (sw) pc += POPCOUNT(sw);
			    }
			    wt = FUZZ1(pc);
			    ACCUM(invar[v0],wt);
			    ACCUM(invar[v1],wt);
                            ACCUM(invar[v2],wt);
                            ACCUM(invar[v3],wt);
                        }
                    }
                }
            }
            wt = invar[lab[cell1]];
            for (i = cell1 + 1; i <= cell2; ++i)
                if (invar[lab[i]] != wt) return;
        }
}

/*****************************************************************************
*                                                                            *
*  distances() assigns to each vertex v a value depending on the number of   *
*  vertices at each distance from v, and what cells they lie in.             *
*  If we find any cell which is split in this manner, we don't try any       *
*  further cells.                                                            *
*                                                                            *
*****************************************************************************/

void
distances(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
          permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i;
        register set *gw;
        register shortish wt;
        int d,dlim,cell1,cell2,iv,v,w;
        boolean success;

#if !MAXN
        DYNALLOC1(set,workset,workset_sz,m,"distances");
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"distances");
	DYNALLOC1(set,ws1,ws1_sz,m,"distances");
	DYNALLOC1(set,ws2,ws2_sz,m,"distances"); 
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        wt = 1;
        for (i = 0; i < n; ++i)
        {
            workshort[lab[i]] = FUZZ1(wt);
            if (ptn[i] <= level) ++wt;
        }

	if (invararg > n || invararg == 0) dlim = n;
	else                               dlim = invararg+1;

        success = FALSE;
        for (cell1 = 0; cell1 < n; cell1 = cell2 + 1)
        {
            for (cell2 = cell1; ptn[cell2] > level; ++cell2) {}
            if (cell2 == cell1) continue;

            for (iv = cell1; iv <= cell2; ++iv)
            {
                v = lab[iv];
                EMPTYSET(ws1,m);
                ADDELEMENT(ws1,v);
                EMPTYSET(ws2,m);
                ADDELEMENT(ws2,v);
                for (d = 1; d < dlim; ++d)
                {
                    EMPTYSET(workset,m);
                    wt = 0;
                    w = -1;
                    while ((w = nextelement(ws2,M,w)) >= 0)
                    {
                        gw = GRAPHROW(g,w,m);
                        ACCUM(wt,workshort[w]);
                        for (i = M; --i >= 0;) workset[i] |= gw[i];
                    }
                    if (wt == 0) break;
                    ACCUM(wt,d);
                    wt = FUZZ2(wt);
                    ACCUM(invar[v],wt);
                    for (i = M; --i >= 0;)
                    {
                        ws2[i] = workset[i] & ~ws1[i];
                        ws1[i] |= ws2[i];
                    }
                }
                if (invar[v] != invar[lab[cell1]]) success = TRUE;
            }
            if (success) break;
        }
}

/*****************************************************************************
*                                                                            *
*  indsets() assigns to each vertex v a value depending on which cells the   *
*  vertices which join v in an independent set lie in.  The size of the      *
*  independent sets which are used is the smallest of invararg and MAXCLIQUE.*
*                                                                            *
*****************************************************************************/

void
indsets(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
        permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i;
        register shortish wt;
        register set *gv;
        int ss,setsize;
        int v[MAXCLIQUE];
        long wv[MAXCLIQUE];
        set *s0,*s1;

#if !MAXN
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"indsets");
	DYNALLOC2(set,wss,wss_sz,m,MAXCLIQUE-1,"indsets");
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        if (invararg <= 1 || digraph) return;

        if (invararg > MAXCLIQUE) setsize = MAXCLIQUE;
        else                      setsize = invararg;

        wt = 1;
        for (i = 0; i < n; ++i)
        {
            workshort[lab[i]] = FUZZ2(wt);
            if (ptn[i] <= level) ++wt;
        }

        for (v[0] = 0; v[0] < n; ++v[0])
        {
            wv[0] = workshort[v[0]];
            s0 = (set*)wss;
            EMPTYSET(s0,m);
            for (i = v[0]+1; i < n; ++i) ADDELEMENT(s0,i);
            gv = GRAPHROW(g,v[0],m);
            for (i = M; --i >= 0;) s0[i] &= ~gv[i];
            ss = 1;
            v[1] = v[0];
            while (ss > 0)
            {
                if (ss == setsize)
                {
                    wt = FUZZ1(wv[ss-1]);
                    for (i = ss; --i >= 0;) ACCUM(invar[v[i]],wt);
                    --ss;
                }
                else if ((v[ss] = nextelement((set*)wss+M*(ss-1),M,v[ss])) < 0)
                    --ss;
                else
                {
                    wv[ss] = wv[ss-1] + workshort[v[ss]];
                    ++ss;
                    if (ss < setsize)
                    {
                        gv = GRAPHROW(g,v[ss-1],m);
			s1 = (set*)wss + M*(ss-2);
                        for (i = M; --i >= 0;) s1[i+M] = s1[i] & ~gv[i];
                        v[ss] = v[ss-1];
                    }
                }
            }
        }
}

/*****************************************************************************
*                                                                            *
*  cliques() assigns to each vertex v a value depending on which cells the   *
*  vertices which join v in a clique lie in.  The size of the cliques used   *
*  is the smallest of invararg and MAXCLIQUE.                                *
*                                                                            *
*****************************************************************************/

void
cliques(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
        permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i;
        register shortish wt;
        register set *gv;
        int ss,setsize;
        int v[MAXCLIQUE];
        long wv[MAXCLIQUE];
	set *ns;

#if !MAXN
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"cliques");
	DYNALLOC2(set,wss,wss_sz,m,MAXCLIQUE-1,"cliques");
#else
	set wss[MAXCLIQUE-1][MAXM];
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        if (invararg <= 1 || digraph) return;

        if (invararg > MAXCLIQUE) setsize = MAXCLIQUE;
        else                      setsize = invararg;

        wt = 1;
        for (i = 0; i < n; ++i)
        {
            workshort[lab[i]] = FUZZ2(wt);
            if (ptn[i] <= level) ++wt;
        }

        for (v[0] = 0; v[0] < n; ++v[0])
        {
            wv[0] = workshort[v[0]];
            gv = GRAPHROW(g,v[0],m);
	    ns = (set*)wss;
            for (i = M; --i >= 0;) ns[i] = gv[i];
            ss = 1;
            v[1] = v[0];
            while (ss > 0)
            {
                if (ss == setsize)
                {
                    wt = FUZZ1(wv[ss-1]);
                    for (i = ss; --i >= 0;) ACCUM(invar[v[i]],wt);
                    --ss;
                }
                else if ((v[ss] = nextelement((set*)wss+M*(ss-1),M,v[ss])) < 0)
                    --ss;
                else
                {
                    wv[ss] = wv[ss-1] + workshort[v[ss]];
                    ++ss;
                    if (ss < setsize)
                    {
                        gv = GRAPHROW(g,v[ss-1],m);
			ns = (set*)wss + M*(ss-2);
                        for (i = M; --i >= 0;) ns[i+M] = ns[i] & gv[i];
                        v[ss] = v[ss-1];
                    }
                }
            }
        }
}

/*****************************************************************************
*                                                                            *
*  cellcliq() assigns to each vertex v a value depending on the number of    *
*  cliques which v lies in and which lie in the same cell as v.  The size    *
*  of clique counted is the smallest of invararg and MAXCLIQUE.  We try the  *
*  cells in increasing order of size and stop as soon as any cell splits.    *
*                                                                            *
*****************************************************************************/

void
cellcliq(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
         permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i;
        register shortish wt;
        register set *gv;
        int ss,setsize;
        int v[MAXCLIQUE];
        set *ns;
        shortish *cellstart,*cellsize;
        int iv,icell,bigcells,cell1,cell2;
        int pc;
        setword sw;

#if !MAXN
        DYNALLOC1(set,workset,workset_sz,m,"cellcliq");
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"cellcliq");
	DYNALLOC2(set,wss,wss_sz,m,MAXCLIQUE-1,"cellcliq");
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        if (invararg <= 1 || digraph) return;

        if (invararg > MAXCLIQUE) setsize = MAXCLIQUE;
        else                      setsize = invararg;

        cellstart = workshort;
        cellsize = workshort + (n/2);
        getbigcells(ptn,level,setsize > 6 ? setsize : 6,&bigcells,
                    cellstart,cellsize,n);

        for (icell = 0; icell < bigcells; ++icell)
        {
            cell1 = cellstart[icell];
            cell2 = cell1 + cellsize[icell] - 1;

            EMPTYSET(workset,m);
            for (iv = cell1; iv <= cell2; ++iv) ADDELEMENT(workset,lab[iv]);

            for (iv = cell1; iv <= cell2; ++iv)
            {
                v[0] = lab[iv];
                gv = GRAPHROW(g,v[0],m);
		ns = (set*)wss;
                pc = 0;

                for (i = M; --i >= 0;)
                {
                    ns[i] = gv[i] & workset[i];
                    if ((sw = ns[i]) != 0) pc += POPCOUNT(sw);
                }
                if (pc <= 1 || pc >= cellsize[icell] - 2) continue;

                ss = 1;
                v[1] = v[0];
                while (ss > 0)
                {
                    if (ss == setsize)
                    {
                        for (i = ss; --i >= 0;) ++invar[v[i]];
                        --ss;
                    }
                    else if ((v[ss] 
				= nextelement((set*)wss+M*(ss-1),M,v[ss])) < 0)
                        --ss;
                    else
                    {
                        ++ss;
                        if (ss < setsize)
                        {
                            gv = GRAPHROW(g,v[ss-1],m);
			    ns = (set*)wss + M*(ss-2);
                            for (i = M; --i >= 0;) ns[i+M] = ns[i] & gv[i];
                            v[ss] = v[ss-1];
                        }
                    }
                }
            }
            wt = invar[lab[cell1]];
            for (iv = cell1; iv <= cell2; ++iv)
                if (invar[lab[i]] != wt) return;
        }
}

/*****************************************************************************
*                                                                            *
*  cellind() assigns to each vertex v a value depending on the number of     *
*  independent sets which v lies in and which lie in the same cell as v.     *
*  The size of clique counted is the smallest of invararg and MAXCLIQUE.     *
*  We try the cells in increasing order of size and stop as soon as any      *
*  cell splits.                                                              *
*                                                                            *
*****************************************************************************/

void
cellind(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
        permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i;
        register shortish wt;
        register set *gv;
        int ss,setsize;
        int v[MAXCLIQUE];
        set *ns;
        shortish *cellstart,*cellsize;
        int iv,icell,bigcells,cell1,cell2;
        int pc;
        setword sw;

#if !MAXN
        DYNALLOC1(set,workset,workset_sz,m,"cellind");
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"cellind");
	DYNALLOC2(set,wss,wss_sz,m,MAXCLIQUE-1,"cellind");
#endif

        for (i = n; --i >= 0;) invar[i] = 0;

        if (invararg <= 1 || digraph) return;

        if (invararg > MAXCLIQUE) setsize = MAXCLIQUE;
        else                      setsize = invararg;

        cellstart = workshort;
        cellsize = workshort + (n/2);
        getbigcells(ptn,level,setsize > 6 ? setsize : 6,&bigcells,
                    cellstart,cellsize,n);

        for (icell = 0; icell < bigcells; ++icell)
        {
            cell1 = cellstart[icell];
            cell2 = cell1 + cellsize[icell] - 1;

            EMPTYSET(workset,m);
            for (iv = cell1; iv <= cell2; ++iv) ADDELEMENT(workset,lab[iv]);

            for (iv = cell1; iv <= cell2; ++iv)
            {
                v[0] = lab[iv];
                gv = GRAPHROW(g,v[0],m);
	 	ns = (set*)wss;
                pc = 0;

                for (i = M; --i >= 0;)
                {
                    ns[i] = ~gv[i] & workset[i];
                    if ((sw = ns[i]) != 0) pc += POPCOUNT(sw);
                }
                if (pc <= 1 || pc >= cellsize[icell] - 2) continue;

                ss = 1;
                v[1] = v[0];
                while (ss > 0)
                {
                    if (ss == setsize)
                    {
                        for (i = ss; --i >= 0;) ++invar[v[i]];
                        --ss;
                    }
                    else if ((v[ss] 
			   = nextelement((set*)wss+M*(ss-1),M,v[ss])) < 0)
                        --ss;
                    else
                    {
                        ++ss;
                        if (ss < setsize)
                        {
                            gv = GRAPHROW(g,v[ss-1],m);
			    ns = (set*)wss + M*(ss-2);
                            for (i = M; --i >= 0;) ns[i+M] = ns[i] & ~gv[i];
                            v[ss] = v[ss-1];
                        }
                    }
                }
            }
            wt = invar[lab[cell1]];
            for (iv = cell1; iv <= cell2; ++iv)
                if (invar[lab[i]] != wt) return;
        }
}

/*****************************************************************************
*                                                                            *
*  adjacencies() assigns to each vertex v a code depending on which cells    *
*  it is joined to and from, and how many times.  It is intended to provide  *
*  better partitioning that the normal refinement routine for digraphs.      *
*                                                                            *
*****************************************************************************/

void
adjacencies(graph *g, int *lab, int *ptn, int level, int numcells, int tvpos,
            permutation *invar, int invararg, boolean digraph, int m, int n)
{
        register int i,v,w;
        register shortish vwt,wwt;
        set *gv;

#if !MAXN
        DYNALLOC1(shortish,workshort,workshort_sz,n+2,"adjacencies");
#endif

        vwt = 1;
        for (i = 0; i < n; ++i)
        {
            workshort[lab[i]] = vwt;
            if (ptn[i] <= level) ++vwt;
            invar[i] = 0;
        }

        for (v = 0, gv = (set*)g; v < n; ++v, gv += M)
        {
            vwt = FUZZ1(workshort[v]);
            wwt = 0;
            w = -1;
            while ((w = nextelement(gv,M,w)) >= 0)
            {
                ACCUM(wwt,FUZZ2(workshort[w]));
                ACCUM(invar[w],vwt);
            }
            ACCUM(invar[v],wwt);
        }
}

/*****************************************************************************
*                                                                            *
*  nautinv_check() checks that this file is compiled compatibly with the     *
*  given parameters.   If not, call exit(1).                                 *
*                                                                            *
*****************************************************************************/

void
nautinv_check(int wordsize, int m, int n, int version)
{
        if (wordsize != WORDSIZE)
        {
            fprintf(ERRFILE,"Error: WORDSIZE mismatch in nautinv.c\n");
            exit(1);
        }

#if MAXN
        if (m > MAXM)
        {
            fprintf(ERRFILE,"Error: MAXM inadequate in nautinv.c\n");
            exit(1);
        }

        if (n > MAXN)
        {
            fprintf(ERRFILE,"Error: MAXN inadequate in nautinv.c\n");
            exit(1);
        }
#endif

#ifdef BIGNAUTY
        if ((version & 1) == 0)
        {   
            fprintf(ERRFILE,"Error: BIGNAUTY mismatch in nautinv.c\n");
            exit(1);
        }
#else
        if ((version & 1) == 1)
        {   
            fprintf(ERRFILE,"Error: BIGNAUTY mismatch in nautinv.c\n");
            exit(1);
        }
#endif

        if (version < NAUTYREQUIRED)
        {
            fprintf(ERRFILE,"Error: nautinv.c version mismatch\n");
            exit(1);
        }
}

/*****************************************************************************
*                                                                            *
*  nautinv_freedyn() - free the dynamic memory in this module                *
*                                                                            *
*****************************************************************************/
 
void
nautinv_freedyn(void)
{
#if !MAXN
	DYNFREE(workset,workset_sz);
	DYNFREE(workshort,workshort_sz);
	DYNFREE(ws1,ws1_sz);
	DYNFREE(ws2,ws2_sz);
	DYNFREE(vv,vv_sz);
	DYNFREE(ww,ww_sz);
	DYNFREE(w01,w01_sz);
	DYNFREE(w02,w02_sz);
	DYNFREE(w03,w03_sz);
	DYNFREE(w12,w12_sz);
	DYNFREE(w13,w13_sz);
	DYNFREE(w23,w23_sz);
	DYNFREE(pt0,pt0_sz);
	DYNFREE(pt1,pt1_sz);
	DYNFREE(pt2,pt2_sz);
	DYNFREE(wss,wss_sz);
#endif
}


syntax highlighted by Code2HTML, v. 0.9.1