ftp.nice.ch/pub/next/science/mathematics/gap.3.4.2.NIHS.bs.tar.gz#/gap.pkg/_gap/lib/gap-3.4.2/src/tietze.c

This is tietze.c in view mode; [Download] [Up]

/****************************************************************************
**
*A  tietze.c                    GAP source                     Volkmar Felsch
**
*A  @(#)$Id: tietze.c,v 3.1 1992/11/16 18:57:36 martin Rel $
**
*Y  Copyright 1990-1992,  Lehrstuhl D fuer Mathematik,  RWTH Aachen,  Germany
**
**  This file contains the functions for computing with finite presentations.
**
*H  $Log: tietze.c,v $
*H  Revision 3.1  1992/11/16  18:57:36  martin
*H  initial revision under RCS
*H
*/

#include        "system.h"              /* system dependent functions      */
#include        "gasman.h"              /* dynamic storage manager         */
#include        "eval.h"                /* evaluator main dispatcher       */
#include        "integer.h"             /* arbitrary size integers         */
#include        "list.h"                /* 'LEN_LIST' macro                */
#include        "agcollec.h"            /* 'HD_WORDS' macro for T_SWORDS   */
#include        "word.h"                /* words                           */

#include        "tietze.h"              /* declaration part of the package */


/****************************************************************************
**
**  Definig some constants for the Tietze routines.
*/
#define    TZ_NUMGENS        1
#define    TZ_NUMRELS        2
#define    TZ_TOTAL          3
#define    TZ_GENERATORS     4
#define    TZ_INVERSES       5
#define    TZ_RELATORS       6
#define    TZ_LENGTHS        7
#define    TZ_FLAGS          8
#define    TZ_LENGTHTIETZE  20


/****************************************************************************
**
*F  TzRelExponent1( <relator> ) . . . . find the exponent of a Tietze relator
**
**  'TzRelExponent1'  determines the exponent of the given relator, i. e. the
**  maximal integer n such that the relator can be expresses as an nth power.
*/
TypHandle       TzRelExponent1 ( hdRel )
    TypHandle           hdRel;
{
    TypHandle           * ptRel;        /* pointer to the Tietze relator   */
    long                leng;           /* length of the given relator     */
    long                exp;            /* exponent of the relator         */
    long                i, j, ij;       /* loop variables                  */

    /*  Initialize some variables                                          */
    ptRel = PTR( hdRel ) + 1;
    leng = HD_TO_INT( ptRel[-1] );
    exp = 1;

    /*  Run through the relator and check for repitions                    */
    for ( i = 1; i <= leng/2; i++ ) {
        if ( leng % i == 0 ) {
            for ( j = 0, ij = i ; j < leng; j++, ij = ( ij + 1 ) % leng ) {
                if ( ptRel[j] != ptRel[ij] )  break;
            }
            if ( j == leng ) {
                exp = leng / i;
                break;  /*  loop over i  */
            }
        }
    }

    return INT_TO_HD( exp );
}


/****************************************************************************
**
*F  FunTzRelator( <hdCall> ) . . . . . . . convert a word to a Tietze relator
**
**  'FunTzRelator'  converts a  word in the group generators  to a Tietze re-
**  lator,  i.e. a free and cyclically reduced word in the Tietze generators.
**  It returns 'HdFalse" if it cannot convert the given word.
*/
TypHandle       FunTzRelator ( hdCall )
    TypHandle           hdCall;
{
    TypHandle           hdTie;          /* handle of the Tietze stack      */
    TypHandle           * ptTie;        /* pointer to the Tietze stack     */
    TypHandle           hdGens;         /* handle of the Tietze gens list  */
    TypHandle           * ptGens;       /* pointer to this list            */
    TypHandle           hdInvs;         /* handle of inverses list         */
    TypHandle           * ptInvs;       /* pointer to this list            */
    TypHandle           hdWrd;          /* handle of the given word        */
    TypHandle           * ptWrd;        /* pointer to the given word       */
    TypHandle           hdRel;          /* handle of the Tietze relator    */
    TypHandle           * ptRel;        /* pointer to the Tietze relator   */
    TypHandle           * pt1, * pt2;   /* pointers to the Tietze relator  */
    TypHandle           hdFac, hdGen;   /* handles of generators           */
    long                numgens;        /* number of Tietze generators     */
    long                leng;           /* length of the given word        */
    long                reduced;        /* length of reduced relator       */
    long                i, j;           /* generator numbers               */

    /* Get and check arguments                                             */
    if ( SIZE(hdCall) != 3*SIZE_HD )
        return Error( "usage: TzRelator(<Tietze stack>,<word>)", 0L, 0L );

    /*  Check the first argument (Tietze stack)                            */
    hdTie = EVAL( PTR(hdCall)[1] );
    if (TYPE(hdTie) != T_LIST || HD_TO_INT(PTR(hdTie)[0]) != TZ_LENGTHTIETZE)
        return Error( "invalid <Tietze stack>", 0L, 0L );
    ptTie = PTR( hdTie );

    /*  Get and check the Tietze generators list                           */
    hdGens = ptTie[TZ_GENERATORS];
    numgens = HD_TO_INT(ptTie[TZ_NUMGENS]);
    if ( hdGens == 0
      || (TYPE(hdGens) != T_LIST && TYPE(hdGens) != T_SET)
      || HD_TO_INT(PTR(hdGens)[0]) != numgens )
        return Error( "invalid Tietze generators list", 0L, 0L );
    ptGens = PTR( hdGens );
    for ( i = 1; i <= numgens; i++ ) {
        if ( TYPE(ptGens[i]) == T_SWORD ) ptGens[i] = WordSword( ptGens[i] );
    }

    /*  Get and check the Tietze inverses list                             */
    hdInvs = ptTie[TZ_INVERSES];
    if ( hdInvs == 0
      || (TYPE(hdInvs) != T_LIST && TYPE(hdInvs) != T_VECTOR)
      || HD_TO_INT(PTR(hdInvs)[0]) != 2 * numgens + 1 )
        return Error( "invalid Tietze inverses list", 0L, 0L );

    /*  Check the second argument (word to be converted)                   */
    hdWrd = EVAL( PTR(hdCall)[2] );
    if ( TYPE(hdWrd) == T_SWORD ) hdWrd = WordSword( hdWrd );
    if ( TYPE(hdWrd) != T_WORD )
        return Error( "TzRelator: <word> must be a word", 0L, 0L );
    leng = SIZE( hdWrd ) / SIZE_HD;

    /*  Allocate a bag for the Tietze relator                              */
    hdRel = NewBag( T_LIST, (leng + 1) * SIZE_HD );
    ptGens = PTR( hdGens );
    ptInvs = PTR( hdInvs ) + numgens + 1;
    ptWrd = PTR( hdWrd );
    ptRel = PTR( hdRel );
    pt2 = ptRel;

    /*  Run through the word, identify each factor in the list of          */
    /*  generators, and convert it.                                        */
    for ( j = 0; j < leng; j++ ) {
        if ( TYPE(ptWrd[j]) == T_SWORD ) ptWrd[j] = WordSword( ptWrd[j] );
        hdFac = ptWrd[j];
        for ( i = 1; i <= numgens; i++ ) {
            hdGen = PTR( ptGens[i] )[0];
            if ( hdFac == hdGen || hdFac == *PTR( hdGen ) )
                 break;
        }
        if ( i > numgens )
            return HdFalse;

        if ( hdFac != hdGen ) i = HD_TO_INT( ptInvs[i] );
        if ( pt2 > ptRel && *pt2 == ptInvs[i] )
            --pt2;
        else
            *++pt2 = INT_TO_HD( i );
    }

    /*  Now cyclically reduce the resulting relator                        */
    pt1 = ++ptRel;
    while ( pt1 < pt2 && *pt1 == ptInvs[HD_TO_INT(*pt2)] ) {
        ++pt1;  --pt2;
    }
    if ( ptRel < pt1 ) {
        while ( pt1 <= pt2 )   *ptRel++ = *pt1++;
        pt2 = --ptRel;
    }

    /*  Resize the resulting relator, if necessary, and return it          */
    reduced = pt2 - PTR( hdRel );
    if ( reduced < leng ) {
        Resize( hdRel, ( reduced + 1 ) * SIZE_HD );
        leng = reduced;
    }

    PTR( hdRel )[0] = INT_TO_HD( leng );
    return hdRel;
}


/****************************************************************************
**
*F  FunTzWord( <hdCall> ) . . . . . . . .  convert a Tietze relator to a word
**
**  'FunTzWord'  converts a Tietze relator to a word in the group generators.
*/
TypHandle       FunTzWord ( hdCall )
    TypHandle           hdCall;
{
    TypHandle           hdTie;          /* handle of the Tietze stack      */
    TypHandle           * ptTie;        /* pointer to the Tietze stack     */
    TypHandle           hdGens;         /* handle of the Tietze gens list  */
    TypHandle           * ptGens;       /* pointer to this list            */
    TypHandle           hdRel;          /* handle of the Tietze relator    */
    TypHandle           * ptRel;        /* pointer to the Tietze relator   */
    TypHandle           hdWrd;          /* handle of the new word          */
    TypHandle           * ptWrd;        /* pointer to the new word         */
    long                numgens;        /* number of Tietze generators     */
    long                leng;           /* length of the Tietze relator    */
    long                i, iabs, j;     /* integer variables               */

    /* Get and check arguments                                             */
    if ( SIZE(hdCall) != 3*SIZE_HD )
        return Error( "usage: TzWord(<Tietze stack>,<Tietze relator>)",
            0L, 0L );

    /*  Check the first argument (Tietze stack)                            */
    hdTie = EVAL( PTR(hdCall)[1] );
    if (TYPE(hdTie) != T_LIST || HD_TO_INT(PTR(hdTie)[0]) != TZ_LENGTHTIETZE)
        return Error( "invalid <Tietze stack>", 0L, 0L );
    ptTie = PTR( hdTie );

    /*  Get and check the Tietze generators                                */
    hdGens = ptTie[TZ_GENERATORS];
    numgens = HD_TO_INT(ptTie[TZ_NUMGENS]);
    if ( hdGens == 0
      || (TYPE(hdGens) != T_LIST && TYPE(hdGens) != T_SET)
      || HD_TO_INT(PTR(hdGens)[0]) != numgens )
        return Error( "invalid Tietze generators list", 0L, 0L );

    /*  Check the second argument (Tietze relator to be converted)         */
    hdRel = EVAL( PTR(hdCall)[2] );
    if ( TYPE(hdRel) != T_LIST && TYPE(hdRel) != T_SET && TYPE(hdRel) !=
        T_VECTOR ) return Error( "invalid <Tietze relator>", 0L, 0L );
    ptRel = PTR( hdRel );
    leng = HD_TO_INT(ptRel[0]);

    /*  Allocate a bag for the new word                                    */
    hdWrd = NewBag( T_WORD, leng * SIZE_HD );
    ptGens = PTR( hdGens );
    ptRel = PTR( hdRel );
    ptWrd = PTR( hdWrd );

    /*  Now loop over all factors and convert them                         */
    for ( j = 0; j < leng; j++ ) {
        i = HD_TO_INT( *++ptRel );
	if ( i < -numgens || numgens < i || !i )
            return Error( "invalid <Tietze relator> entry [%d] = %d", j, i );
        iabs = ( i > 0 ) ? i : -i;
        if ( TYPE(ptGens[iabs]) == T_SWORD )
            ptGens[iabs] = WordSword( ptGens[iabs] );
        if ( TYPE(ptGens[iabs]) != T_WORD || SIZE(ptGens[iabs]) != SIZE_HD )
            return Error( "invalid Tietze generator [%d]", iabs, 0L );
        *ptWrd++ = ( i > 0 ) ?
            PTR( ptGens[iabs] )[0] : *PTR( PTR( ptGens[iabs] )[0] );
    }

    return hdWrd;
}


/****************************************************************************
**
*F  FunTzSortC(<hdCall>)  . . . . . . . . . . . . sort the relators by length
*/
TypHandle       FunTzSortC ( hdCall )
    TypHandle           hdCall;
{
    TypHandle           hdTie;          /* handle of the Tietze stack      */
    TypHandle           * ptTie;        /* pointer to the Tietze stack     */
    TypHandle           hdRels;         /* handle of the relators list     */
    TypHandle           * ptRels;       /* pointer to this list            */
    TypHandle           hdLens;         /* handle of the lengths list      */
    TypHandle           * ptLens;       /* pointer to this list            */
    TypHandle           hdFlags;        /* handle of the flags list        */
    TypHandle           * ptFlags;      /* pointer to this list            */
    unsigned long       numrels;        /* number of Tietze relators       */
    unsigned long       i, h, k;        /* loop variables                  */
    TypHandle           hdrel, hdlen;   /* handles of list entries         */
    TypHandle           hdflag;         /* handle of a list entry          */

    /* Get and check arguments                                             */
    if ( SIZE(hdCall) != 2*SIZE_HD )
        return Error( "usage: TzSortC(<Tietze stack>)", 0L, 0L );

    /*  Check the Tietze stack                                             */
    hdTie = EVAL( PTR(hdCall)[1] );
    if (TYPE(hdTie) != T_LIST || HD_TO_INT(PTR(hdTie)[0]) != TZ_LENGTHTIETZE)
        return Error( "invalid <Tietze stack>", 0L, 0L );
    ptTie = PTR( hdTie );

    /*  Get and check the Tietze relators list                             */
    hdRels = ptTie[TZ_RELATORS];
    numrels = HD_TO_INT(ptTie[TZ_NUMRELS]);
    if ( hdRels == 0
      || (TYPE(hdRels) != T_LIST && TYPE(hdRels) != T_SET)
      || HD_TO_INT(PTR(hdRels)[0]) != numrels )
        return Error( "invalid Tietze relators list", 0L, 0L );
    ptRels = PTR( hdRels );

    /*  Get and check the Tietze lengths list                              */
    hdLens = ptTie[TZ_LENGTHS];
    if ( hdLens == 0
      || (TYPE(hdLens) != T_VECTOR && TYPE(hdLens) != T_LIST)
      || HD_TO_INT(PTR(hdLens)[0]) != numrels )
        return Error( "invalid Tietze lengths list", 0L, 0L );
    ptLens = PTR( hdLens );

    /*  Get and check the Tietze flags list                                */
    hdFlags = ptTie[TZ_FLAGS];
    if ( hdFlags == 0
      || (TYPE(hdFlags) != T_VECTOR && TYPE(hdFlags) != T_LIST)
      || HD_TO_INT(PTR(hdFlags)[0]) != numrels )
        return Error( "invalid Tietze flags list", 0L, 0L );
    ptFlags = PTR( hdFlags );

    /* Check list <lengths> to contain the relator lengths                 */
    for ( i = 1; i <= numrels; i++ ) {
        if ( ptRels[i] == 0
          || (TYPE(ptRels[i]) != T_LIST && TYPE(ptRels[i]) != T_SET
           && TYPE(ptRels[i]) != T_VECTOR)
          || HD_TO_INT(ptLens[i]) != HD_TO_INT(PTR(ptRels[i])[0]) )
            return Error( "inconsistent Tietze lengths list", 0L, 0L );
    }

    h = 1;
    while ( 9 * h + 4 < numrels ) h = 3 * h + 1;
    while ( 0 < h ) {
        for ( i = h + 1; i <= numrels; i++ ) {
            hdrel = ptRels[i];  hdlen = ptLens[i];  hdflag = ptFlags[i];
            k = i;
            if ( HD_TO_INT(hdlen) ) {
                while ( h < k
                    && ( !HD_TO_INT(ptLens[k-h])
                       || hdlen < ptLens[k-h]
                       || (hdlen == ptLens[k-h] && hdflag > ptFlags[k-h]))) {
                    ptRels[k] = ptRels[k-h];
                    ptLens[k] = ptLens[k-h];
                    ptFlags[k] = ptFlags[k-h];
                    k = k - h;
                }
            }
            ptRels[k] = hdrel;  ptLens[k] = hdlen;  ptFlags[k] = hdflag;
        }
        h = h / 3;
    }
    for ( i = numrels; i > 0; i-- ) {
        if ( HD_TO_INT(ptLens[i]) )
            break;
    }
    if ( i < numrels ) {
        ptRels[0] = ptLens[0] = ptFlags[0] = ptTie[TZ_NUMRELS] =
            INT_TO_HD ( i );
        Resize( hdRels, ( i + 1 ) * SIZE_HD );
        Resize( hdLens, ( i + 1 ) * SIZE_HD );
        Resize( hdFlags, ( i + 1 ) * SIZE_HD );
    }

    return HdVoid;
}


/****************************************************************************
**
*F  FunTzRenumberGens(<hdCall>)  . . . . . . . renumber the Tietze generators
*/
TypHandle       FunTzRenumberGens ( hdCall )
    TypHandle           hdCall;
{
    TypHandle           hdTie;          /* handle of the Tietze stack      */
    TypHandle           * ptTie;        /* pointer to this stack           */
    TypHandle           hdRels;         /* handle of the relators list     */
    TypHandle           * ptRels;       /* pointer to this list            */
    TypHandle           hdInvs;         /* handle of the inverses list     */
    TypHandle           * ptInvs;       /* pointer to this list            */
    TypHandle           * ptRel;        /* pointer to the ith relator      */
    long                numgens;        /* number of Tietze generators     */
    long                numrels;        /* number of Tietze relators       */
    long                old;            /* generator or inverse            */
    long                leng;           /* relator length                  */
    long                i, j;           /* loop variables                  */

    /*  Get and check arguments                                            */
    if ( SIZE(hdCall) != 2*SIZE_HD )
        return Error( "usage: TzRenumberGens(<Tietze stack>)", 0L, 0L );

    /*  Check the Tietze stack                                             */
    hdTie = EVAL( PTR(hdCall)[1] );
    if (TYPE(hdTie) != T_LIST || HD_TO_INT(PTR(hdTie)[0]) != TZ_LENGTHTIETZE)
        return Error( "invalid <Tietze stack>", 0L, 0L );
    ptTie = PTR( hdTie );

    /*  Get and check the Tietze relators list                             */
    hdRels = ptTie[TZ_RELATORS];
    numrels = HD_TO_INT(ptTie[TZ_NUMRELS]);
    if ( hdRels == 0
      || (TYPE(hdRels) != T_LIST && TYPE(hdRels) != T_SET)
      || HD_TO_INT(PTR(hdRels)[0]) != numrels )
        return Error( "invalid Tietze relators list", 0L, 0L );
    ptRels = PTR( hdRels );

    /*  Get and check the Tietze inverses list                             */
    hdInvs = ptTie[TZ_INVERSES];
    numgens = HD_TO_INT(ptTie[TZ_NUMGENS]);
    if ( hdInvs == 0
      || (TYPE(hdInvs) != T_LIST && TYPE(hdInvs) != T_VECTOR)
      || HD_TO_INT(PTR(hdInvs)[0]) != 2 * numgens + 1 )
        return Error( "invalid Tietze inverses list", 0L, 0L );
    ptInvs = PTR( hdInvs ) + numgens + 1;

    /*  Loop over all relators and replace the occurring generators        */
    for ( i = 1; i <= numrels; i++ ) {
        ptRel = PTR( ptRels[i] );
        leng = HD_TO_INT( ptRel[0] );
        /* Run through the relator and replace the occurring generators    */
        for ( j = 1; j <= leng; j++ ) {
            old = HD_TO_INT( ptRel[j] );
            if ( old < -numgens || numgens < old || old == 0 )
                return Error( "gen no. %d in rel no. %d out of range", j,i );
            ptRel[j] = ptInvs[-old];
        }
    }

    return HdVoid;
}


/****************************************************************************
**
*F  FunTzReplaceGens(<hdCall>)  . . . replace Tietze generators by other ones
*/
TypHandle       FunTzReplaceGens ( hdCall )
    TypHandle           hdCall;
{
    TypHandle           hdTie;          /* handle of the Tietze stack      */
    TypHandle           * ptTie;        /* pointer to this stack           */
    TypHandle           hdRels;         /* handle of the relators list     */
    TypHandle           * ptRels;       /* pointer to this list            */
    TypHandle           hdLens;         /* handle of the lengths list      */
    TypHandle           * ptLens;       /* pointer to this list            */
    TypHandle           hdFlags;        /* handle of the flags list        */
    TypHandle           * ptFlags;      /* pointer to this list            */
    TypHandle           hdInvs;         /* handle of the inverses list     */
    TypHandle           * ptInvs;       /* pointer to this list            */
    TypHandle           hdRel;          /* handle of a relator             */
    TypHandle           * ptRel;        /* pointer to this relator         */
    TypHandle           * pt1, * pt2;   /* pointers to a relator           */
    long                numgens;        /* number of Tietze generators     */
    long                numrels;        /* number of Tietze relators       */
    long                total;          /* total length of relators        */
    long                old, new;       /* generators or inverses          */
    long                leng, reduced;  /* relator lengths                 */
    long                altered;        /* flag                            */
    long                i, j;           /* loop variables                  */

    /*  Get and check arguments                                            */
    if ( SIZE(hdCall) != 2*SIZE_HD )
        return Error( "usage: TzReplaceGens(<Tietze stack>)", 0L, 0L );

    /*  Check the Tietze stack                                             */
    hdTie = EVAL( PTR(hdCall)[1] );
    if (TYPE(hdTie) != T_LIST || HD_TO_INT(PTR(hdTie)[0]) != TZ_LENGTHTIETZE)
        return Error( "invalid <Tietze stack>", 0L, 0L );
    ptTie = PTR( hdTie );

    /*  Get and check the Tietze relators list                             */
    hdRels = ptTie[TZ_RELATORS];
    numrels = HD_TO_INT(ptTie[TZ_NUMRELS]);
    if ( hdRels == 0
      || (TYPE(hdRels) != T_LIST && TYPE(hdRels) != T_SET)
      || HD_TO_INT(PTR(hdRels)[0]) != numrels )
        return Error( "invalid Tietze relators list", 0L, 0L );
    ptRels = PTR( hdRels );

    /*  Get and check the Tietze lengths list                              */
    hdLens = ptTie[TZ_LENGTHS];
    if ( hdLens == 0
      || (TYPE(hdLens) != T_VECTOR && TYPE(hdLens) != T_LIST)
      || HD_TO_INT(PTR(hdLens)[0]) != numrels )
        return Error( "invalid Tietze lengths list", 0L, 0L );
    ptLens = PTR( hdLens );

    /*  Check list <lengths> to contain the relator lengths                */
    total = 0;
    for ( i = 1; i <= numrels; i++ ) {
        if ( ptRels[i] == 0
          || (TYPE(ptRels[i]) != T_LIST && TYPE(ptRels[i]) != T_SET
           && TYPE(ptRels[i]) != T_VECTOR)
          || HD_TO_INT(ptLens[i]) != HD_TO_INT(PTR(ptRels[i])[0]) )
            return Error( "inconsistent Tietze lengths list entry [%d]",
               i, 0L );
        total += HD_TO_INT(ptLens[i]);
    }
    if ( total != HD_TO_INT(ptTie[TZ_TOTAL]) )
        return Error( "inconsistent total length", 0L, 0L );

    /*  Get and check the Tietze flags list                                */
    hdFlags = ptTie[TZ_FLAGS];
    if ( hdFlags == 0
      || (TYPE(hdFlags) != T_VECTOR && TYPE(hdFlags) != T_LIST)
      || HD_TO_INT(PTR(hdFlags)[0]) != numrels )
        return Error( "invalid Tietze flags list", 0L, 0L );
    ptFlags = PTR( hdFlags );

    /*  Get and check the Tietze inverses list                             */
    hdInvs = ptTie[TZ_INVERSES];
    numgens = HD_TO_INT(ptTie[TZ_NUMGENS]);
    if ( hdInvs == 0
      || (TYPE(hdInvs) != T_LIST && TYPE(hdInvs) != T_VECTOR)
      || HD_TO_INT(PTR(hdInvs)[0]) != 2 * numgens + 1 )
        return Error( "invalid Tietze inverses list", 0L, 0L );
    ptInvs = PTR( hdInvs ) + numgens + 1;

    /*  Loop over all relators                                             */
    for ( i = 1; i <= numrels; i++ ) {
        hdRel = ptRels[i];
        pt2 = ptRel = PTR( hdRel );
        leng = HD_TO_INT( ptLens[i] );
        altered = 0;

        /* Don't change a sqare relator defining a valid involution        */
        if ( HD_TO_INT( ptFlags[i] ) == 3 && leng == 2 &&
            ptRel[1] == ptInvs[-HD_TO_INT(ptRel[1])] ) {
            continue;  /*  loop over i  */
        }

        /* Run through the relator and replace the occurring generators    */
        for ( j = 1; j <= leng; j++ ) {
            old = HD_TO_INT( ptRel[j] );
            if ( old < -numgens || numgens < old || old == 0 )
                return Error( "gen no. %d in rel no. %d out of range", j,i );
            new = HD_TO_INT( ptInvs[-old] );
            if ( !new ) {
                altered = 1;
                continue;  /*  loop over j  */
            }

            if ( pt2 > ptRel && *pt2 == ptInvs[new] ) {
                altered = 1;
                --pt2;
            }
            else {
                if ( new != old )  { altered = 1; }
                *++pt2 = INT_TO_HD( new );
            }
        }

        if ( !altered ) { continue; }  /*  loop over i  */

        /*  Now cyclically reduce the relator                              */
        pt1 = ++ptRel;
        while ( pt1 < pt2 && *pt1 == ptInvs[HD_TO_INT(*pt2)] ) {
            ++pt1;  --pt2;
        }
        if ( ptRel < pt1 ) {
            while ( pt1 <= pt2 )  { *ptRel++ = *pt1++; }
            pt2 = --ptRel;
        }

        /*  Resize the resulting relator, if necessary                     */
        ptRel = PTR( hdRel );
        reduced = pt2 - ptRel;
        if ( reduced < leng ) {
            ptRel[0] = INT_TO_HD( reduced );
            ptLens[i] = INT_TO_HD( reduced );
            total = total - leng + reduced;
            Resize( hdRel, ( reduced + 1 ) * SIZE_HD );
            ptRels = PTR( hdRels );
            ptLens = PTR( hdLens );
            ptFlags = PTR( hdFlags );
            ptInvs = PTR( hdInvs ) + numgens + 1;
        }

        /*  Redefine the corresponding search flag                         */
        PTR( hdFlags )[i] = INT_TO_HD( 1 );
    }

    ptTie = PTR( hdTie );
    ptTie[TZ_TOTAL] = INT_TO_HD( total );

    return HdVoid;
}


/****************************************************************************
**
*F  FunTzSubstituteGen(<hdCall>)  . . replace a Tietze generator by some word
*/
TypHandle       FunTzSubstituteGen ( hdCall )
    TypHandle           hdCall;
{
    TypHandle           hdTie;          /* handle of the Tietze stack      */
    TypHandle           * ptTie;        /* pointer to this stack           */
    TypHandle           hdRels;         /* handle of the relators list     */
    TypHandle           * ptRels;       /* pointer to this list            */
    TypHandle           hdLens;         /* handle of the lengths list      */
    TypHandle           * ptLens;       /* pointer to this list            */
    TypHandle           hdFlags;        /* handle of the flags list        */
    TypHandle           hdInvs;         /* handle of the inverses list     */
    TypHandle           * ptInvs;       /* pointer to this list            */
    TypHandle           hdWrd;          /* handle of the replacing word    */
    TypHandle           * ptWrd;        /* pointer to this word            */
    TypHandle           hdIwrd;         /* handle of the inverse word      */
    TypHandle           * ptIwrd;       /* pointer to this word            */
    TypHandle           hdNew;          /* handle of a modified relator    */
    TypHandle           * ptNew;        /* pointer to this relator         */
    TypHandle           hdRel;          /* handle of a relator             */
    TypHandle           * ptRel;        /* pointer to this relator         */
    TypHandle           * pt1, * pt2;   /* pointers to a relator           */
    TypHandle           * pt3;          /* pointer to a relator            */
    long                numgens;        /* number of Tietze generators     */
    long                numrels;        /* number of Tietze relators       */
    long                total;          /* total length of relators        */
    long                given;          /* given generator and inverse     */
    long                gen, ginv;      /* given generator and inverse     */
    long                next;           /* generator or inverse            */
    long                leng, newleng;  /* relator lengths                 */
    long                wleng;          /* length of the replacing word    */
    long                occ;            /* number of occurrences           */
    long                i, j;           /* loop variables                  */

    /*  Get and check arguments                                            */
    if ( SIZE(hdCall) != 4*SIZE_HD ) return Error( 
        "usage: TzSubstituteGen(<Tietze stack>,<gen no.>,<Tietze word>)",
        0L, 0L );

    /*  Check the first argument (Tietze stack)                            */
    hdTie = EVAL( PTR(hdCall)[1] );
    if (TYPE(hdTie) != T_LIST || HD_TO_INT(PTR(hdTie)[0]) != TZ_LENGTHTIETZE)
        return Error( "invalid <Tietze stack>", 0L, 0L );
    ptTie = PTR( hdTie );

    /*  Get and check the Tietze relators list                             */
    hdRels = ptTie[TZ_RELATORS];
    numrels = HD_TO_INT(ptTie[TZ_NUMRELS]);
    if ( hdRels == 0
      || (TYPE(hdRels) != T_LIST && TYPE(hdRels) != T_SET)
      || HD_TO_INT(PTR(hdRels)[0]) != numrels )
        return Error( "invalid Tietze relators list", 0L, 0L );
    ptRels = PTR( hdRels );

    /*  Get and check the Tietze lengths list                              */
    hdLens = ptTie[TZ_LENGTHS];
    if ( hdLens == 0
      || (TYPE(hdLens) != T_VECTOR && TYPE(hdLens) != T_LIST)
      || HD_TO_INT(PTR(hdLens)[0]) != numrels )
        return Error( "invalid Tietze lengths list", 0L, 0L );
    ptLens = PTR( hdLens );

    /*  Get and check the Tietze flags list                                */
    hdFlags = ptTie[TZ_FLAGS];
    if ( hdFlags == 0
      || (TYPE(hdFlags) != T_VECTOR && TYPE(hdFlags) != T_LIST)
      || HD_TO_INT(PTR(hdFlags)[0]) != numrels )
        return Error( "invalid Tietze flags list", 0L, 0L );

    /*  Get and check the inverses list                                    */
    hdInvs = ptTie[TZ_INVERSES];
    numgens = HD_TO_INT(ptTie[TZ_NUMGENS]);
    if ( hdInvs == 0
      || (TYPE(hdInvs) != T_LIST && TYPE(hdInvs) != T_VECTOR)
      || HD_TO_INT(PTR(hdInvs)[0]) != 2 * numgens + 1 )
        return Error( "invalid Tietze inverses list", 0L, 0L );
    ptInvs = PTR( hdInvs ) + numgens + 1;

    /*  Check the second argument (generator number)                       */
    given = HD_TO_INT( EVAL( PTR(hdCall)[2] ) );
    gen = ( given > 0 ) ? given : -given;
    if ( gen <= 0 || numgens < gen )
        return Error( "generator number %d out of range", gen, 0L );
    ginv = HD_TO_INT(ptInvs[gen]);

    /*  Check the third argument (replacing word)                          */
    hdWrd = EVAL( PTR(hdCall)[3] );
    if ( TYPE(hdWrd) != T_LIST && TYPE(hdWrd) != T_VECTOR )
        return Error( "invalid replacing word", 0L, 0L );
    ptWrd = PTR( hdWrd );
    wleng = HD_TO_INT( ptWrd[0] );
    for ( i = 1; i <= wleng; i++ ) {
        next = HD_TO_INT( ptWrd[i] );
        if ( next < -numgens || next == 0 || next > numgens )
            return Error( "entry [%d] of <Tietze word> out of range", i,0L );
    }

    /*  Check list <lengths> to contain the relator lengths                */
    total = 0;
    for ( i = 1; i <= numrels; i++ ) {
        if ( ptRels[i] == 0
          || (TYPE(ptRels[i]) != T_LIST && TYPE(ptRels[i]) != T_SET
           && TYPE(ptRels[i]) != T_VECTOR)
          || HD_TO_INT(ptLens[i]) != HD_TO_INT(PTR(ptRels[i])[0]) )
            return Error( "inconsistent Tietze lengths list", 0L, 0L );
        total += HD_TO_INT(ptLens[i]);
    }
    if ( total != HD_TO_INT(ptTie[TZ_TOTAL]) )
        return Error( "inconsistent total length", 0L, 0L );

    /*  Allocate a bag for the inverse of the replacing word               */
    hdIwrd = NewBag( T_LIST, (wleng + 1) * SIZE_HD );
    ptRels = PTR( hdRels );
    ptLens = PTR( hdLens );
    ptInvs = PTR( hdInvs ) + numgens + 1;
    ptWrd = PTR( hdWrd );
    ptIwrd = PTR( hdIwrd );

    /*  Invert the replacing word                                          */
    ptIwrd[0] = INT_TO_HD( wleng );
    pt1 = ptWrd;  pt2 = ptIwrd + wleng;
    while ( pt2 > ptIwrd )
        *pt2-- = ptInvs[HD_TO_INT(*++pt1)];
    if ( given < 0 ) {
        hdNew = hdWrd;  hdWrd = hdIwrd;  hdIwrd = hdNew;
        ptWrd = PTR( hdWrd );  ptIwrd = PTR( hdIwrd );
    }

    /*  Loop over all relators                                             */
    for ( i = 1; i <= numrels; i++ ) {
        hdRel = ptRels[i];
        ptRel = PTR( hdRel );
        leng = HD_TO_INT( ptLens[i] );
        if ( leng == 0 )  { continue; }

        /* Run through the relator and count the occurrences of gen        */
        occ = 0;
        for ( j = 1; j <= leng; j++ ) {
            next = HD_TO_INT( ptRel[j] );
            if ( next < -numgens || numgens < next )
                return Error( "gen no. %d in rel no. %d out of range", j,i );
            if (next == gen || next == ginv ) ++occ;
        }
        if ( occ == 0 )  { continue; }

        /*  Allocate a bag for the modified Tietze relator                 */
        hdNew = NewBag( T_LIST, (leng + occ * (wleng - 1) + 1) * SIZE_HD );
        ptLens = PTR( hdLens );
        ptInvs = PTR( hdInvs ) + numgens + 1;
        ptWrd = PTR( hdWrd );
        ptIwrd = PTR( hdIwrd );
        ptRel = PTR( hdRel );
        pt2 = ptNew = PTR( hdNew );

        /*  Now run again through the relator and modify it                */
        for ( j = 1; j <= leng; j++ ) {
            next = HD_TO_INT( ptRel[j] );
            if ( next == gen || next == -gen ) {
                pt1 = ( next > 0 ) ? ptWrd : ptIwrd;
                pt3 = pt1 + wleng;
                while ( pt1 < pt3 ) {
                    ++pt1;
                    if ( pt2 > ptNew && *pt2 == ptInvs[HD_TO_INT(*pt1)] )
                        --pt2;
                    else
                        *++pt2 = *pt1;
                }
            }
            else {
                if ( pt2 > ptNew && *pt2 == ptInvs[next] )
                    --pt2;
                else
                    *++pt2 = INT_TO_HD( next );
            }
        }

        /*  Now cyclically reduce the relator                              */
        pt1 = ++ptNew;
        while ( pt1 < pt2 && *pt1 == ptInvs[HD_TO_INT(*pt2)] ) {
            ++pt1;  --pt2;
        }
        if ( ptNew < pt1 ) {
            while ( pt1 <= pt2 )   *ptNew++ = *pt1++;
            pt2 = --ptNew;
        }

        /*  Resize and save the resulting relator                          */
        ptNew = PTR( hdNew );
        newleng = pt2 - ptNew;
        ptNew[0] = INT_TO_HD( newleng );
        ptLens[i] = INT_TO_HD( newleng );
        total = total - leng + newleng;
        Resize( hdNew, ( newleng + 1 ) * SIZE_HD );
        ptRels = PTR( hdRels );
        ptLens = PTR( hdLens );
        ptRels[i] = hdNew;
        PTR( hdFlags )[i] = INT_TO_HD( 1 );
    }

    ptTie = PTR( hdTie );
    ptTie[TZ_TOTAL] = INT_TO_HD( total );

    return HdVoid;
}


/****************************************************************************
**
*F  FunTzOccurrences( <hdCall> ) . . . . . . occurrences of Tietze generators
**
**  'FunTzOccurrences' implements the internal function 'TzOccurrences'.
*/
TypHandle       FunTzOccurrences ( hdCall )
    TypHandle           hdCall;
{
    TypHandle           hdTie;          /* handle of the Tietze stack      */
    TypHandle           * ptTie;        /* pointer to the Tietze stack     */
    TypHandle           hdRels;         /* handle of the relators list     */
    TypHandle           * ptRels;       /* pointer to this list            */
    TypHandle           hdRes;          /* handle of the result            */
    TypHandle           hdCnts;         /* list of the counts              */
    TypHandle           * ptCnts;       /* pointer to the counts list      */
    TypHandle           hdMins;         /* list of minimal occurence list  */
    TypHandle           * ptMins;       /* pointer to the minimals list    */
    TypHandle           hdLens;         /* list of lengths of those        */
    TypHandle           * ptLens;       /* pointer to the lengths list     */
    TypHandle           hdRel;          /* handle of a relator             */
    TypHandle           * ptRel;        /* pointer to this relator         */
    TypHandle           hdAux;          /* auxiliary list                  */
    long                * ptAux;        /* pointer to the lengths list     */
    long                numgens;        /* number of Tietze generators     */
    long                numrels;        /* number of Tietze relators       */
    long                leng;           /* length of a relator             */
    long                num, next;      /* generators or inverses          */
    long                i, k;           /* loop variables                  */
    long                c;              /* count of one generator          */
    long                nr;             /* number of occurrences           */
    long                nr1;            /* nr of occurrences in one word   */
    long                nrm;            /* minimal value of 'nr1'          */
    long                min;            /* word that has this minimum      */

    /* Get and check arguments                                             */
    if ( SIZE(hdCall) != 2*SIZE_HD && SIZE(hdCall) != 3*SIZE_HD )
        return Error("usage: TzOccurrences( <Tietze stack> [, <gen no.> ] )",
            0L, 0L );

    /*  Check the first argument (Tietze stack)                            */
    hdTie = EVAL( PTR(hdCall)[1] );
    if (TYPE(hdTie) != T_LIST || HD_TO_INT(PTR(hdTie)[0]) != TZ_LENGTHTIETZE)
        return Error( "invalid <Tietze stack>", 0L, 0L );
    ptTie = PTR( hdTie );

    /*  Get and check the Tietze relators list                             */
    numgens = HD_TO_INT(ptTie[TZ_NUMGENS]);
    numrels = HD_TO_INT(ptTie[TZ_NUMRELS]);
    hdRels = ptTie[TZ_RELATORS];
    if ( hdRels == 0
      || (TYPE(hdRels) != T_LIST && TYPE(hdRels) != T_SET)
      || HD_TO_INT(PTR(hdRels)[0]) != numrels )
        return Error( "invalid Tietze relators list", 0L, 0L );
    ptRels = PTR( hdRels );

    /*  Get and check the given generator number                           */
    if ( SIZE(hdCall) == 3*SIZE_HD ) {
        num = HD_TO_INT( EVAL( PTR(hdCall)[2] ) );
        if ( num <= 0 || numgens < num )
            return Error( "given generator number out of range", 0L, 0L );
        numgens = 1;
    }
    else  num = numgens;

    /* allocate the result lists                                           */
    hdCnts = NewBag( T_LIST, SIZE_HD + numgens * SIZE_HD );
    PTR(hdCnts)[0] = INT_TO_HD( numgens );
    for ( k = 1; k <= numgens; k++ )
        PTR(hdCnts)[k] = INT_TO_HD( 0 );
    hdMins = NewBag( T_LIST, SIZE_HD + numgens * SIZE_HD );
    PTR(hdMins)[0] = INT_TO_HD( 0 );
    hdLens = NewBag( T_LIST, SIZE_HD + numgens * SIZE_HD );
    PTR(hdLens)[0] = INT_TO_HD( 0 );
    hdRes = NewBag( T_LIST, SIZE_HD + 3 * SIZE_HD );
    PTR(hdRes)[0] = INT_TO_HD( 3 );
    PTR(hdRes)[1] = hdCnts;
    PTR(hdRes)[2] = hdMins;
    PTR(hdRes)[3] = hdLens;

    /* allocate an auxiliary list                                          */
    ptAux = 0;
    if ( numgens > 1 ) {
        hdAux = NewBag( T_STRING, ( numgens + 1 ) * sizeof( long ) );
        ptAux = (long*)PTR(hdAux);
        ptAux[0] = numgens;
        for ( k = 1; k <= numgens; k++ )
            ptAux[k] = 0;

    }

    /* now we can safely grab pointers                                     */
    ptRels = PTR(hdRels);
    ptCnts = PTR(hdCnts);
    ptLens = PTR(hdLens);
    ptMins = PTR(hdMins);

    /* handle special case of single generator in generator list           */
    if ( numgens == 1 ) {

        /* initialize the counters                                         */
        nr = 0;  nrm = 0;  min = 0;

        /* loop over all relators                                          */
        for ( i = 1; i <= numrels; i++ ) {
            hdRel = ptRels[i];
            if ( hdRel == 0
              || (TYPE(hdRel) != T_LIST && TYPE(hdRel) != T_SET
               && TYPE(hdRel) != T_VECTOR) )
                return Error("invalid entry [%d] in Tietze relators list",
                             i,0L);
            ptRel = PTR(hdRel);
            leng = HD_TO_INT( ptRel[0] );

            /* loop over the letters of the relator                        */
            nr1 = 0;
            for ( k = 1; k <= leng; k++ ) {
                next = HD_TO_INT( ptRel[k] );
                if ( next == num || next == -num )  { nr1++; }
            }

            /* check whether the number of occurrences of num is less than */
            /* in the preceding relators                                   */
            nr += nr1;
            if ( nrm == 0
              || (0 < nr1 && nr1 < nrm)
              || (nr1 == nrm && SIZE(hdRel) < SIZE(ptRels[min])) ) {
                nrm = nr1;  min = i;
            }
        }

        /* put the information into the result bags                        */
        ptCnts[1] = INT_TO_HD( nr );
        if ( nr != 0 ) {
            ptCnts[1] = INT_TO_HD( nr );
            ptLens[0] = INT_TO_HD( 1 );  ptLens[1] = INT_TO_HD( nrm );
            ptMins[0] = INT_TO_HD( 1 );  ptMins[1] = INT_TO_HD( min );
        }
    }

    /* handle general case of all Tietze generators                        */
    else {

        /* loop over all relators                                          */
        for ( i = 1; i <= numrels; i++ ) {
            hdRel = ptRels[i];
            if ( hdRel == 0
              || (TYPE(hdRel) != T_LIST && TYPE(hdRel) != T_SET
               && TYPE(hdRel) != T_VECTOR) )
                return Error("invalid entry [%d] in Tietze relators list",
                             i,0L);
            ptRel = PTR(hdRel);
            leng = HD_TO_INT( ptRel[0] );

            /* loop over the letters of the relator                        */
            for ( k = 1; k <= leng; k++ ) {
                next = HD_TO_INT( ptRel[k] );
                if ( next < 0 ) next = -next;
                if ( next == 0 || numgens < next ) return Error(
                    "invalid entry [%d][%d] in Tietze relators list",i,k );
                (ptAux[next])++;
            }

            /* loop over the generators, collecting the counts             */
            for ( k = 1; k <= numgens; k++ ) {
                c = ptAux[k];
                if ( !c )  continue;
                ptAux[k] = 0;
                ptCnts[k] = INT_TO_HD( HD_TO_INT( ptCnts[k] ) + c );
                if ( (0 < c && ptLens[k] == 0)
                  || (0 < c && c < HD_TO_INT(ptLens[k]))
                  || (0 < c && c == HD_TO_INT(ptLens[k])
                    && SIZE(hdRel) < SIZE(ptRels[HD_TO_INT(ptMins[k])])) ) {
                    ptLens[k] = INT_TO_HD( c );
                    ptMins[k] = INT_TO_HD( i );
                }
            }
        }

        /* find the correct length of the minimals and lengths lists       */
        k = numgens;
        while ( ptMins[k] == 0 )  k--;
        PTR(hdMins)[0] = INT_TO_HD( k );
        PTR(hdLens)[0] = INT_TO_HD( k );
    }

    /* return the result                                                   */
    return hdRes;
}


/****************************************************************************
**
*F  FunTzOccurrencesPairs(<hdCall>) occurrences of pairs of Tietze generators
*/
TypHandle       FunTzOccurrencesPairs ( hdCall )
    TypHandle           hdCall;
{
    TypHandle           hdTie;          /* handle of the Tietze stack      */
    TypHandle           * ptTie;        /* pointer to the Tietze stack     */
    TypHandle           hdRels;         /* handle of the relators list     */
    TypHandle           * ptRels;       /* pointer to this list            */
    TypHandle           hdInvs;         /* handle of the inverses list     */
    TypHandle           * ptInvs;       /* pointer to this list            */
    TypHandle           hdRes;          /* handle of the resulting list    */
    TypHandle           * ptRes;        /* pointer to this list            */
    TypHandle           hdRel;          /* handle of a relator             */
    TypHandle           * ptRel;        /* pointer to this relator         */
    TypHandle           hdNum;          /* handle of generator number      */
    TypHandle           hdInv;          /* handle of inverse gen number    */
    long                num, i, ii;     /* generator numbers               */
    long                numgens;        /* number of Tietze generators     */
    long                numrels;        /* number of Tietze relators       */
    long                length;         /* length of the current relator   */
    long                j1, j2, r;      /* loop variables                  */

    /* Get and check arguments                                             */
    if ( SIZE(hdCall) != 3*SIZE_HD && SIZE(hdCall) != 4*SIZE_HD ) return
        Error(
            "usage: TzOccurrencesPairs( <Tietze stack>, <gen> [, <list> ] )",
            0L, 0L );

    /*  Check the first argument (Tietze stack)                            */
    hdTie = EVAL( PTR(hdCall)[1] );
    if (TYPE(hdTie) != T_LIST || HD_TO_INT(PTR(hdTie)[0]) != TZ_LENGTHTIETZE)
        return Error( "invalid <Tietze stack>", 0L, 0L );
    ptTie = PTR( hdTie );

    /*  Get and check the Tietze relators list                             */
    hdRels = ptTie[TZ_RELATORS];
    numrels = HD_TO_INT(ptTie[TZ_NUMRELS]);
    if ( hdRels == 0
      || (TYPE(hdRels) != T_LIST && TYPE(hdRels) != T_SET)
      || HD_TO_INT(PTR(hdRels)[0]) != numrels )
        return Error( "invalid Tietze relators list", 0L, 0L );

    /*  Get and check the Tietze inverses list                             */
    hdInvs = ptTie[TZ_INVERSES];
    numgens = HD_TO_INT(ptTie[TZ_NUMGENS]);
    if ( hdInvs == 0
      || (TYPE(hdInvs) != T_LIST && TYPE(hdInvs) != T_VECTOR)
      || HD_TO_INT(PTR(hdInvs)[0]) != 2 * numgens + 1 )
        return Error( "invalid Tietze inverses list", 0L, 0L );

    /*  Get and check the Tietze generator number                          */
    hdNum = EVAL( PTR(hdCall)[2] );
    if ( TYPE(hdNum) != T_INT )
        return Error( "<gen> must be a Tietze generator number", 0L, 0L );
    num = HD_TO_INT( hdNum );
    if ( num <= 0 || num > numgens )
        return Error( "given generator number is out of range", 0L, 0L );

    /*  Get and check the list for the results, if specified               */
    if ( SIZE(hdCall) == 3* SIZE_HD ) {
        hdRes = NewBag( T_LIST, SIZE_HD + 4 * numgens * SIZE_HD );
        PTR( hdRes )[0] = INT_TO_HD( 4 * numgens );
    }
    else {
        hdRes = EVAL( PTR(hdCall)[3] );
        if ( hdRes == 0
          || (TYPE(hdRes) != T_LIST && TYPE(hdRes) != T_VECTOR)
          || HD_TO_INT(PTR(hdRes)[0]) != 4 * numgens )
            return Error(
                "<list> must be a list of length 4 * number of generators",
                         0L, 0L );
    }

    /*  return, if num = numgens                                           */
    if ( num == numgens )  { return hdRes; }

    /*  get pointers to the involved lists                                 */
    ptRels = PTR( hdRels );
    ptInvs = PTR( hdInvs ) + numgens + 1;
    ptRes = PTR( hdRes );

    /* get the handle of the inverse of the given generator                */
    hdInv = ptInvs[num];

    /* ptRes[i]           counts the occurrences of gen * gen[i]           */
    /* ptRes[numgens+i]   counts the occurrences of gen * gen[i]^-1        */
    /* ptRes[2*numgens+i] counts the occurrences of gen^-1 * gen[i]        */
    /* ptRes[3*numgens+i] counts the occurrences of gen^-1 * gen[i]^-1     */

    /* initialize the counters                                             */
    for ( i = 1; i <= 4 * numgens; i++ ) {
        ptRes[i] = INT_TO_HD( 0 );
    }

    /* loop over the Tietze relators                                       */
    for ( r = 1; r <= numrels; r++ ) {
        hdRel = ptRels[r];
        if ( hdRel == 0
          || (TYPE(hdRel) != T_LIST && TYPE(hdRel) != T_SET
           && TYPE(hdRel) != T_VECTOR) )
            return Error( "invalid Tietze relator [%d]", 0L, 0L );
        ptRel = PTR( hdRel ) + 1;

        /* skip the current relator if its length is less than 2           */
        length = HD_TO_INT( ptRel[-1] );
        if ( length < 2 )  { continue; }

        /* loop over the current relator and investigate the pairs         */
        /* ( ptRel[j1], ptRel[j2] )                                        */
        j1 = ( length - 1 ) % length;
        for ( j2 = 0; j2 < length; j1 = j2, j2++ ) {

            /* count any "forward" pair  gen * gen[i],  gen * gen[i]^-1,   */
            /* gen^-1 * gen[i],  or  gen^-1 * gen[i]^-1  ( with num < i )  */
            if ( ptRel[j1] == hdNum || ptRel[j1] == hdInv ) {
                i = HD_TO_INT( ptRel[j2] );
                if ( -num <= i && i <= num )  { continue; }
	        if ( i < - numgens || numgens < i ) return Error(
                    "invalid entry %d in <Tietze relator> [%d]", i, r );
                if ( i < 0 ) i = numgens - i;
                if ( ptRel[j1] != hdNum ) i = i + 2 * numgens;
                ptRes[i] = INT_TO_HD( HD_TO_INT( ptRes[i] ) + 1 );
            }

            /* count any "backward" pair  gen[i]^-1 * gen^-1,              */
            /* gen[i] * gen^-1,  gen[i]^-1 * gen,  or  gen[i] * gen        */
            /* ( with num < i )  which is not covered by a forward pair    */
            else if ( ptRel[j2] == hdNum || ptRel[j2] == hdInv ) {
                i = HD_TO_INT( ptRel[j1] );
                if ( -num <= i && i <= num )  { continue; }
	        if ( i < - numgens || numgens < i ) return Error(
                    "invalid entry %d in <Tietze relator> [%d]", i, r );
                ii = HD_TO_INT( ptInvs[i] );
                if ( !( (hdNum == hdInv
                        && ptRel[(j2+1)%length] == INT_TO_HD(ii))
                     || (i == ii
                        && ptInvs[HD_TO_INT(ptRel[(j1+length-1)%length])] 
                           == ptRel[j2]) ) ) {
                    if ( ii < 0 ) ii = numgens - ii;
                    if ( ptRel[j2] != hdInv ) ii = ii + 2 * numgens;
                    ptRes[ii] = INT_TO_HD( HD_TO_INT( ptRes[ii] ) + 1 );
                }
            }
        }
    }

    return hdRes;
}


/****************************************************************************
**
*F  FunTzSearchC(<hdCall>) . . . . .  find subword matches in Tietze relators
**
**  'FunTzSearchC' implements the internal function 'TzSearchC'.
*/
TypHandle       FunTzSearchC ( hdCall )
    TypHandle           hdCall;
{
    TypHandle           hdTie;          /* handle of the Tietze stack      */
    TypHandle           * ptTie;        /* pointer to this stack           */
    TypHandle           hdRels;         /* handle of the relators list     */
    TypHandle           * ptRels;       /* pointer to this list            */
    TypHandle           hdLens;         /* handle of the lengths list      */
    TypHandle           * ptLens;       /* pointer to this list            */
    TypHandle           hdInvs;         /* handle of the inverses list     */
    TypHandle           * ptInvs;       /* pointer to this list            */
    TypHandle           hdFlags;        /* handle of the flags list        */
    TypHandle           * ptFlags;      /* pointer to this list            */
    TypHandle           hdWrd;          /* handle of the given relator     */
    TypHandle           hdl;            /* handle of current list relator  */
    TypHandle           hdw;            /* handle of a relator             */
    TypHandle           hdPos1;         /* handle of the second argument   */
    TypHandle           hdPos2;         /* handle of the third argument    */
    TypHandle           hdEqu;          /* handle of the fifth argument    */
    char                keys1[8192], keys2[8192], keys3[8192];
                                        /* hash table of key values        */
    unsigned long       inv;            /* inverse for key computation     */
    unsigned long       key;            /* key value of subword            */
    long                numgens;        /* number of Tietze generators     */
    long                numrels;        /* number of Tietze relators       */
    long                total;          /* total length of relators        */
    TypHandle           * ptr;          /* pointer to a relator            */
    TypHandle           * v,  * w;      /* pointers into relators          */
    TypHandle           * ptx,  * pty;  /* pointers into relators          */
    long                i1, j1;         /* loop variables                  */
    long                i2, j2;         /* loop variables                  */
    long                i3, j3;         /* loop variables                  */
    long                len1;           /* relator length                  */
    long                lmin, lmax;     /* bound for relator lengths       */
    long                pos1, pos2;     /* position of the given relator   */
    long                xmax;           /* position of the given relator   */
    long                newflag, flag1; /* Tietze relator flags            */
    long                xflag, yflag;   /* Tietze relator flags            */
    long                xlen, xlen1;    /* length of the given relator     */
    long                mlen;           /* length of the wanted match      */
    long                ylen, ylen1;    /* length of the current relator   */
    long                newlen;         /* length of a new relator         */
    long                n, m;           /* subword lengths                 */
    long                count;          /* number of altered relators      */
    long                i, j, jj, x, y; /* loop variables                  */
    long                lasty;          /* flag                            */
    long                altered;        /* flag                            */
    long                equal;          /* flag                            */

    /* Get and check arguments                                             */
    if ( SIZE(hdCall) != 4*SIZE_HD && SIZE(hdCall) != 5*SIZE_HD )
        return Error(
            "usage: TzSearchC( <Tietze stack>, <pos1>, <pos1> [,<equal>] )",
            0L, 0L );

    /*  Check the first argument (Tietze stack)                            */
    hdTie = EVAL( PTR(hdCall)[1] );
    if (TYPE(hdTie) != T_LIST || HD_TO_INT(PTR(hdTie)[0]) != TZ_LENGTHTIETZE)
        return Error( "invalid <Tietze stack>", 0L, 0L );
    ptTie = PTR( hdTie );

    /*  Get and check the Tietze relators list                             */
    hdRels = ptTie[TZ_RELATORS];
    numrels = HD_TO_INT(ptTie[TZ_NUMRELS]);
    if ( hdRels == 0
      || (TYPE(hdRels) != T_LIST && TYPE(hdRels) != T_SET)
      || HD_TO_INT(PTR(hdRels)[0]) != numrels )
        return Error( "invalid Tietze relators list", 0L, 0L );
    ptRels = PTR( hdRels );

    /*  Get and check the Tietze lengths list                              */
    hdLens = ptTie[TZ_LENGTHS];
    if ( hdLens == 0
      || (TYPE(hdLens) != T_VECTOR && TYPE(hdLens) != T_LIST)
      || HD_TO_INT(PTR(hdLens)[0]) != numrels )
        return Error( "invalid Tietze lengths list", 0L, 0L );
    ptLens = PTR( hdLens );

    /*  Get and check the Tietze flags list                                */
    hdFlags = ptTie[TZ_FLAGS];
    if ( hdFlags == 0
      || (TYPE(hdFlags) != T_VECTOR && TYPE(hdFlags) != T_LIST)
      || HD_TO_INT(PTR(hdFlags)[0]) != numrels )
        return Error( "invalid Tietze flags list", 0L, 0L );
    ptFlags = PTR( hdFlags );

    /*  Check list <lengths> to contain the relator lengths                */
    total = 0;
    for ( i = 1; i <= numrels; i++ ) {
        if ( ptRels[i] == 0
          || (TYPE(ptRels[i]) != T_LIST && TYPE(ptRels[i]) != T_SET
           && TYPE(ptRels[i]) != T_VECTOR)
          || HD_TO_INT(ptLens[i]) != HD_TO_INT(PTR(ptRels[i])[0]) )
            return Error( "inconsistent Tietze lengths list entry [%d]",
                         i, 0L );
        total += HD_TO_INT(ptLens[i]);
    }
    if ( total != HD_TO_INT(ptTie[TZ_TOTAL]) )
        return Error( "inconsistent total length", 0L, 0L );

    /*  Get and check the Tietze inverses list                             */
    hdInvs = ptTie[TZ_INVERSES];
    numgens = HD_TO_INT(ptTie[TZ_NUMGENS]);
    if ( hdInvs == 0
      || (TYPE(hdInvs) != T_LIST && TYPE(hdInvs) != T_VECTOR)
      || HD_TO_INT(PTR(hdInvs)[0]) != 2 * numgens + 1 )
        return Error( "invalid Tietze inverses list", 0L, 0L );
    ptInvs = PTR( hdInvs ) + numgens + 1;

    /*  Check the second argument                                          */
    hdPos1 = EVAL( PTR(hdCall)[2] );
    if ( TYPE(hdPos1) != T_INT )
       return Error( "<position1> must be a posititve int", 0L ,0L );
    pos1 = HD_TO_INT( hdPos1 );
    if ( pos1 > numrels )
       return Error( "<position1> not in range: %d", pos1, 0L );

    /*  Check the third argument                                           */
    hdPos2 = EVAL( PTR(hdCall)[3] );
    if ( TYPE(hdPos2) != T_INT )
       return Error( "<position2> must be a posititve int", 0L, 0L );
    pos2 = HD_TO_INT( hdPos2 );
    if ( pos2 > numrels )
       return Error( "<position2> not in range: %d", pos2, 0L );

    /*  Check the fourth argument                                          */
    hdEqu = (SIZE(hdCall)==4*SIZE_HD) ? HdFalse : EVAL(PTR(hdCall)[4]);
    if ( TYPE(hdEqu) != T_BOOL )
       return Error( "<equal> must be false or true", 0L, 0L );
    equal = ( hdEqu == HdTrue );

    /*  Skip relators of inconvenient lengths or with inconvenient flags,  */
    /*  and return if the remaining range is empty                         */
    while ( pos1 <= pos2
        && (HD_TO_INT( ptLens[pos1] ) < 2
         || HD_TO_INT( ptFlags[pos1] ) > 1
         || (equal && ( HD_TO_INT( ptLens[pos1] ) < 4
                     || HD_TO_INT( ptLens[pos1] ) % 2 == 1 ) ) ) )
        pos1++;
    if ( pos1 > pos2 || pos1 == numrels )  { return INT_TO_HD( 0 ); }

    /*  Get the range of compatible relator lengths                        */
    len1 = HD_TO_INT( ptLens[pos1] );
    lmin = len1 - ( len1 % 2 );
    lmax = ( equal ) ? lmin : lmin + 1;

    /*  Initialize some variables                                          */
    newflag = ( equal ) ? 1 : 2;
    count = 0;
    lasty = 0;
    xmax = pos1 - 1;
    flag1 = HD_TO_INT( ptFlags[pos1] );

    /* Compute the length of the wanted match and the corresponding        */
    /* inverse factor                                                      */
    mlen = equal ? ( lmin + 1 ) / 2 : lmin / 2 + 1;
    inv = 1;
    for ( i = 1; i <= mlen; i++ )
       inv = 109109 * inv;

    /* Initialize the hash table                                           */
    for ( i = 0; i < 2048; i++ )
       ((unsigned long *)keys1)[i] =
       ((unsigned long *)keys2)[i] =
       ((unsigned long *)keys3)[i] = 0;

    /* Loop over the Tietze relators, starting at position pos1            */
    for ( y = pos1; y < numrels; ) {
       hdWrd = ptRels[y];
       ylen = HD_TO_INT( ptLens[y] );
       yflag = HD_TO_INT( ptFlags[y] );
       if ( y <= pos2 && lmin <= ylen && ylen <= lmax && yflag <= 1 ) {

          /* Add the key values of the current relator to the hash table   */
          ptr = PTR(hdWrd);
          key = 0;
          for ( i = 0, w = ptr+1; i < mlen; i++, w++ )
             key = 109109 * key + ( (unsigned long)*w >> 2 );
          for ( i = 0, v = ptr+1, w = v+mlen; i < ylen; i++, v++, w++ ) {
             keys1[ key & 8191 ] = 1;
             keys2[ (key >> 11) & 8191 ] |= (1 << ((key >> 8) & 7));
             keys3[ (key >> 19) & 8191 ] |= (1 << ((key >> 16) & 7));
             if ( i == ylen-mlen )  w = ptr+1;
             key = 109109 * key - inv * ( (unsigned long)*v >> 2 )
                 + ( (unsigned long)*w >> 2 );
          }
          key = 0;
          for ( i = 0, w = ptr+ylen; i < mlen; i++, w-- ) {
             key = 109109 * key
                 + ( (unsigned long) ptInvs[HD_TO_INT(*w)] >> 2 );
          }
          for ( i = 0, v = ptr+ylen, w = v-mlen; i < ylen; i++, v--, w-- ) {
             keys1[ key & 8191 ] = 1;
             keys2[ (key >> 11) & 8191 ] |= (1 << ((key >> 8) & 7));
             keys3[ (key >> 19) & 8191 ] |= (1 << ((key >> 16) & 7));
             if ( i == ylen-mlen )  w = ptr+ylen;
             key = 109109 * key
                 - inv * ( (unsigned long) ptInvs[HD_TO_INT(*v)] >> 2 )
                 + ( (unsigned long) ptInvs[HD_TO_INT(*w)] >> 2 );
          }
          if ( len1 > ylen ) len1 = ylen;
          if ( flag1 < yflag ) flag1 = yflag;
          xmax = y;
       }

       /* Move to next relator                                             */
       y++;

       /*  Initialize some variables                                       */
       hdl = ptRels[y];
       ylen = HD_TO_INT( ptLens[y] );
       yflag = HD_TO_INT( ptFlags[y] );
       ylen1 = ylen - 1;
       altered = 0;

       /* Loop to the next relator, if the current relator is too short    */
       if ( y > lasty
         && (ylen < len1 || yflag > 1 || (!equal && !(yflag + flag1)) ) )
          continue;  /*  loop over y                                       */
       lasty = y;

       /* Compute the key values of the current relator                    */
       ptr = PTR(hdl);
       key = 0;
       for ( j = 0, w = ptr+1; j < mlen; j++, w++ )
          key = 109109 * key + ( (unsigned long)*w >> 2 );
       for ( j = 0; j < ylen; j++ ) {

          /* Check for key match in the tables                             */
          if ( keys1[ key & 8191 ]
             && (keys2[ (key >> 11) & 8191 ] & (1 << ((key >> 8) & 7)))
             && (keys3[ (key >> 19) & 8191 ] & (1 << ((key >> 16) & 7))) ){

             /* Loop over the (relevant) given relators                    */
             for ( x = pos1; x <= xmax; x++ ) {

                hdw = ptRels[x];
                xlen = HD_TO_INT( ptLens[x] );
                xflag = HD_TO_INT( ptFlags[x] );
                if ( xlen < len1 || xlen > lmax || xlen > ylen
                  || xflag > 1 || (!equal && !( xflag + yflag )) )
                   continue;  /*  loop over x                              */

                xlen1 = xlen - 1;
                ptx = PTR(hdw) + 1;
                pty = PTR(hdl) + 1;

                /* Loop over all possible positions in the given relator   */
                for ( i = 0; i < xlen; i++ ) {

                   /* Search forward for a match of length at least mlen   */
                   i2 = i;  j2 = j;
                   for ( n = 0; n < xlen; n++,
                      i2 = (i2 == xlen1) ? 0 : i2 + 1,
                      j2 = (j2 == ylen1) ? 0 : j2 + 1 ) {
                      if ( ptx[i2] != pty[j2] )  break;  /*  loop over n   */
                   }
                   if ( n < mlen )  continue;  /*  loop over i             */

                   /* Search backward to find the whole match              */
                   i1 = (i == 0) ? xlen1 : i - 1;
                   j1 = (j == 0) ? ylen1 : j - 1;
                   for ( ; n < xlen; n++,
                      i1 = (i1 == 0) ? xlen1 : i1 - 1,
                      j1 = (j1 == 0) ? ylen1 : j1 - 1 ) {
                      if ( ptx[i1] != pty[j1] )  break;  /*  loop over n   */
                   }

                   /* Replace a matching substring of equal length         */
                   if ( n == xlen - n ) {
                      j2 = j;
                      for ( m = 0; m < n; m++,
                         i1 = (i1 == 0) ? xlen1 : i1 - 1,
                         j2 = (j2 == ylen1) ? 0 : j2 + 1 ) {
                         pty[j2] = ptInvs[HD_TO_INT(ptx[i1])];
                      }

                      /* Now replace all exact occurrences of this string  */
                      /* in the current word (not relator)                 */
                      i3 = (i + n) % xlen;

                      for ( jj = 0; jj <= ylen - n; jj++ ) {
                         i2 = i;  j2 = jj;
                         for ( m = 0; m < n; m++,
                            i2 = (i2 == xlen1) ? 0 : i2 + 1,
                            j2 = (j2 == ylen1) ? 0 : j2 + 1 ) {
                            if ( ptx[i2] != pty[j2] )
                                break;  /*  loop over m                    */
                         }
                         if ( m < n )  continue;  /*  loop over jj         */

                         i1 = (i == 0) ? xlen1 : i - 1;
                         if ( ptx[i1] == pty[(jj + ylen1) % ylen] ||
                            ptx[i3] == pty[(jj + n) % ylen] )
                            continue;  /*  loop over jj                    */

                         j2 = jj;
                         for ( m = 0; m < n; m++,
                            i1 = (i1 == 0) ? xlen1 : i1 - 1,
                            j2 = (j2 == ylen1) ? 0 : j2 + 1 ) {
                            pty[j2] = ptInvs[HD_TO_INT(ptx[i1])];
                         }

                         jj = -1;
                      }

                      ptFlags[y] = INT_TO_HD( newflag );
                      altered = 1;
                      ++count;
                      break;  /*  loop over i                              */
                   }

                   m = ylen - n;  n = xlen - n;

                   /* Find all canceling factors                           */
                   if ( n == 0 ) {
                      for ( ; 1 < m; m -= 2,
                         j1 = (j1 == 0) ? ylen1 : j1 - 1,
                         j2 = (j2 == ylen1) ? 0 : j2 + 1 ) {
                         if ( pty[j1] != ptInvs[HD_TO_INT(pty[j2])] )
                            break;  /*  loop over m                        */
                      }
                   }

                   /* Create the modified relator and save it              */
                   newlen = m + n;
                   if ( j2 > 0 ) {
                      if ( j2 <= j1 )  { jj = 0;  j3 = j1;  j1 = m - 1; }
                      else  { jj = j1 + n + 1;  j3 = ylen - 1; }
                      for ( ; j2 <= j3; ) {
                         pty[jj++] = pty[j2++];
                      }
                   }
                   for ( ; n > 0; n--, i1 = (i1 == 0) ? xlen1 : i1 - 1 ) {
                      pty[++j1] = ptInvs[HD_TO_INT(ptx[i1])];
                   }
                   pty[-1] = INT_TO_HD( newlen );
                   ptLens[y] = INT_TO_HD( newlen );
                   total = total - ylen + newlen;
                   ptFlags[y] = INT_TO_HD( newflag );

                   /* Reduce the bag size                                  */
                   Resize( hdl, (newlen + 1) * SIZE_HD );
                   ptRels = PTR( hdRels );
                   ptLens = PTR( hdLens );
                   ptFlags = PTR( hdFlags);
                   ptInvs = PTR( hdInvs ) + numgens + 1;

                   altered = 1;
                   ++count;
                   --y;
                   break;  /*  loop over i                                 */
                }

                if ( altered ) break;  /*  loop over x                     */

                /* Now try the inverse of the given relator                */
                for ( i = 0; i < xlen; i++ ) {

                   /* Search forward for a match of length at least mlen   */
                   i2 = xlen1 - i;  j2 = j;
                   for ( n = 0; n < xlen; n++,
                      i2 = (i2 == 0) ? xlen1 : i2 - 1,
                      j2 = (j2 == ylen1) ? 0 : j2 + 1 ) {
                      if ( ptInvs[HD_TO_INT(ptx[i2])] != pty[j2] )
                         break;  /*  loop over n                           */
                   }
                   if ( n < mlen )  continue;  /*  loop over i             */

                   /* Search backward to find the whole match              */
                   i1 = (i == 0) ? 0 : xlen - i;
                   j1 = (j == 0) ? ylen1 : j - 1;
                   for ( ; n < xlen; n++,
                      i1 = (i1 == xlen1) ? 0 : i1 + 1,
                      j1 = (j1 == 0) ? ylen1 : j1 - 1 ) {
                      if ( ptInvs[HD_TO_INT(ptx[i1])] != pty[j1] )
                         break;  /*  loop over n                           */
                   }

                   /* Replace a matching substring of equal length         */
                   if ( n == xlen - n ) {
                      j2 = j;
                      for ( m = 0; m < n; m++,
                         i1 = (i1 == xlen1) ? 0 : i1 + 1,
                         j2 = (j2 == ylen1) ? 0 : j2 + 1 ) {
                         pty[j2] = ptx[i1];
                      }

                      ptFlags[y] = INT_TO_HD( newflag );
                      altered = 1;
                      ++count;
                      break;  /*  loop over i                              */
                   }

                   m = ylen - n;  n = xlen - n;

                   /* Find all canceling factors                           */
                   if ( n == 0 ) {
                      for ( ; 1 < m; m -= 2,
                         j1 = (j1 == 0) ? ylen1 : j1 - 1,
                         j2 = (j2 == ylen1) ? 0 : j2 + 1 ) {
                         if ( pty[j1] != ptInvs[HD_TO_INT(pty[j2])] )
                            break;  /*  loop over m                        */
                      }
                   }

                   /* Create the modified relator and save it              */
                   newlen = m + n;
                   if ( j2 > 0 ) {
                      if ( j2 <= j1 )  { jj = 0;  j3 = j1;  j1 = m - 1; }
                      else  { jj = j1 + n + 1;  j3 = ylen - 1; }
                      for ( ; j2 <= j3; ) {
                         pty[jj++] = pty[j2++];
                      }
                   }
                   for ( ; n > 0; n--, i1 = (i1 == xlen1) ? 0 : i1 + 1 ) {
                      pty[++j1] = ptx[i1];
                   }
                   pty[-1] = INT_TO_HD( newlen );
                   ptLens[y] = INT_TO_HD( newlen );
                   total = total - ylen + newlen;
                   ptFlags[y] = INT_TO_HD( newflag );

                   /* Reduce the bag size                                  */
                   Resize( hdl, (newlen + 1) * SIZE_HD );
                   ptRels = PTR( hdRels );
                   ptLens = PTR( hdLens );
                   ptFlags = PTR( hdFlags);
                   ptInvs = PTR( hdInvs ) + numgens + 1;

                   altered = 1;
                   ++count;
                   --y;
                   break;  /*  loop over i                                 */
                }

                if ( altered ) break;  /*  loop over x                     */
             }
          }

          if ( altered ) break;  /*  loop over j                           */

          v = ptr + 1 + j;  w = ptr + 1 + ( j + mlen ) % ylen;
          key = 109109 * key - inv * ( (unsigned long)*v >> 2 )
              + ( (unsigned long)*w >> 2 );
       }
    }

    PTR( hdTie )[TZ_TOTAL] = INT_TO_HD( total );

    /* Return the number of altered relators                               */
    return INT_TO_HD( count );
}


/****************************************************************************
**
*F  InitTietze()  . . . . . . . . . . . . . . . . . initialize tietze package
**
**  'InitTietze' initializes the Tietze package.
*/
void            InitTietze ()
{
    InstIntFunc( "TzRelator",           FunTzRelator          );
    InstIntFunc( "TzWord",              FunTzWord             );
    InstIntFunc( "TzSortC",             FunTzSortC            );
    InstIntFunc( "TzRenumberGens",      FunTzRenumberGens     );
    InstIntFunc( "TzReplaceGens",       FunTzReplaceGens      );
    InstIntFunc( "TzSubstituteGen",     FunTzSubstituteGen    );
    InstIntFunc( "TzOccurrences",       FunTzOccurrences      );
    InstIntFunc( "TzOccurrencesPairs",  FunTzOccurrencesPairs );
    InstIntFunc( "TzSearchC",           FunTzSearchC          );
}


/****************************************************************************
**
*E  Emacs . . . . . . . . . . . . . . . . . . . . . . . local emacs variables
**
**  Local Variables:
**  mode:               outline
**  outline-regexp:     "*F\\|*V\\|*T\\|*E"
**  fill-column:        73
**  fill-prefix:        "**  "
**  eval:               (local-set-key "\t" 'c-indent-command)
**  eval:               (local-set-key ";"  'electric-c-semi )
**  eval:               (local-set-key "{"  'electric-c-brace)
**  eval:               (local-set-key "}"  'electric-c-brace)
**  eval:               (hide-body)
**  End:
*/



These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.