Quellcodebibliothek Statistik Leitseite products/sources/formale Sprachen/GAP/pkg/ctbllib/tst/   (Algebra von RWTH Aachen Version 4.15.1©)  Datei vom 22.4.2025 mit Größe 88 kB image not shown  

Quelle  ctblbm.tst   Sprache: unbekannt

 
# This file was created from xpl/ctblbm.xpl, do not edit!
#############################################################################
##
#W  ctblbm.tst                GAP applications              Thomas Breuer
##
##  In order to run the tests, one starts GAP from the `tst` subdirectory
##  of the `pkg/ctbllib` directory, and calls `Test( "ctblbm.tst" );`.
##
gap> START_TEST( "ctblbm.tst" );

##
gap> LoadPackage( "ctbllib", false );
true
gap> LoadPackage( "atlasrep", false );
true

##
gap> CTblLib.IsMagmaAvailable();
true

##
gap> F:= FreeGroup( List( "abcdefghijk", x -> [ x ] ) );;
gap> gens:= GeneratorsOfGroup( F );;
gap> rels:= List( gens, x -> x^2 );;
gap> ord3pairs:= List( [ 1 .. 7 ], i -> [ i, i+1 ] );
[ [ 1, 2 ], [ 2, 3 ], [ 3, 4 ], [ 4, 5 ], [ 5, 6 ], [ 6, 7 ], [ 7, 8 ] ]
gap> Append( ord3pairs, [ [ 5, 9 ], [ 9, 10 ], [ 10, 11 ] ] );
gap> for pair in ord3pairs do
>      Add( rels, ( gens[ pair[1] ] * gens[ pair[2] ] )^3 );
>    od;
gap> for i in [ 1 .. 11 ] do
>      for j in [ i+1 .. 11 ] do
>        if not [ i, j ] in ord3pairs then
>          Add( rels, ( gens[i] * gens[j] )^2 );
>        fi;
>      od;
>    od;
gap> Add( rels, Product( gens{ [ 5, 4, 3, 5, 6, 7, 5, 9, 10 ] } )^10 );

##
gap> gensstrings:= List( gens, String );;
gap> relsslps:= List( List( rels, String ),
>                     x -> StraightLineProgram( x, gensstrings ) );;

##
gap> slp:= [ [ 1, 1, 2, 1 ], [ 3, 5 ], [ 4, 3 ], [ 5, 1, 2, 1 ] ];;
gap> resultpos:= [];;
gap> Add( slp, [ 6, 19 ] );
gap> resultpos[11]:= Length( slp ) + 2;;

##
gap> Append( slp, [ [ 1, 1, 7, 1 ], [ 8, 3 ] ] );
gap> Append( slp, [ [ 6, 3 ], [ 9, 1, 10, 1 ], [ 11, 10 ],
>                   [ 6, -1, 12, 1, 6, 1 ] ] );

##
gap> # 14: e*c,  15: (e*c)^2,  16: (e*c)^3,  17: (e*c)^4,  18: (e*c)^6
gap> Append( slp, [ [ 13, 1, 9, 1 ], [ 14, 2 ], [ 14, 1, 15, 1 ],
>                   [ 15, 2 ], [ 16, 2 ] ] );
gap> # 19: e*c*e,  20: e*c*e^2*c
gap> Append( slp, [ [ 14, 1, 13, 1 ], [ 19, 1, 14, 1 ] ] );
gap> # 21: (e*c)^6*c*(e*c)^3,  22: ((e*c)^6*c*(e*c)^3)^2*e*c*e^2*c
gap> Append( slp, [ [ 18, 1, 9, 1, 16, 1 ], [ 21, 2, 20, 1 ] ] );
gap> # 23: (((e*c)^6*c*(e*c)^3)^2*e*c*e^2*c)^5
gap> Append( slp, [ [ 22, 5 ] ] );
gap> # 24: t1 = f = ((((e*c)^6*c*(e*c)^3)^2*e*c*e^2*c)^5)^((e*c)^4)
gap> Append( slp, [ [ 17, -1, 23, 1, 17, 1 ] ] );
gap> resultpos[1]:= Length( slp ) + 2;;

##
gap> # 25: (e*c)^8*c*(e*c)^3,  26: (e*c*e^2*c)^2
gap> Append( slp, [ [ 15, 1, 21, 1 ], [ 20, 2 ] ] );
gap> # 27: g = ((e*c)^8*c*(e*c)^3)^((e*c*e^2*c)^2)
gap> Append( slp, [ [ 26, -1, 25, 1, 26, 1 ] ] );
gap> # 28: g f,  29: g^-1,  30: t2 = f^{{g f}}
gap> Append( slp, [ [ 27, 1, 24, 1 ], [ 27, -1 ],
>                   [ 28, -1, 24, 1, 28, 1 ] ] );
gap> resultpos[2]:= Length( slp ) + 2;;

##
gap> # 31: t3 = f^( g * f * g )
gap> Append( slp, [ [ 29, 1, 30, 1, 27, 1 ] ] );
gap> resultpos[3]:= Length( slp ) + 2;;
gap> # 32: t4 = f^( g * f * g^2 )
gap> Append( slp, [ [ 29, 1, 31, 1, 27, 1 ] ] );
gap> resultpos[4]:= Length( slp ) + 2;;
gap> # 33: t5 = f^( g * f * g^3 )
gap> Append( slp, [ [ 29, 1, 32, 1, 27, 1 ] ] );
gap> resultpos[5]:= Length( slp ) + 2;;
gap> # 34: t6 = f^( g * f * g^4 )
gap> Append( slp, [ [ 29, 1, 33, 1, 27, 1 ] ] );
gap> resultpos[6]:= Length( slp ) + 2;;
gap> # 35: t7 = f^( g * f * g^5 )
gap> Append( slp, [ [ 29, 1, 34, 1, 27, 1 ] ] );
gap> resultpos[7]:= Length( slp ) + 2;;
gap> # 36: t8 = f^( g * f * g^6 )
gap> Append( slp, [ [ 29, 1, 35, 1, 27, 1 ] ] );
gap> resultpos[8]:= Length( slp ) + 2;;

##
gap> # 37: (a*b)^5*t11*(a*b)^-5*t1*(a*b)^5,
gap> # 38: p = ((a*b)^5*t11*(a*b)^-5*t1*(a*b)^5)^-1
gap> Append( slp, [ [ 4, 1, 7, 1, 4, -1, 24, 1, 4, 1 ], [ 37, -1 ] ] );
gap> # 39: p^-1,  40: h = c^p, 41: i = d^p
gap> Append( slp, [ [ 38, -1 ], [ 39, 1, 9, 1, 38, 1 ],
>                   [ 39, 1, 6, 1, 38, 1 ] ] );

##
gap> # 42: i^2,  43: t5^(i^2),  44: t3*t4
gap> Append( slp, [ [ 41, 2 ], [ 42, -1, 33, 1, 42, 1 ],
>                   [ 31, 1, 32, 1 ] ] );
gap> # 45: j = Comm( t5^( i^2 ), t3*t4 )
gap> Append( slp, [ [ 43, -1, 44, -1, 43, 1, 44, 1 ] ] );
gap> # 46: i^3,  47: t5^(i^5),  48: k = Comm( t5^(i^5), t3*t4 )
gap> Append( slp, [ [ 41, 1, 42, 1 ], [ 46, -1, 43, 1, 46, 1 ],
>                   [ 47, -1, 44, -1, 47, 1, 44, 1 ] ] );

##
gap> # 49: t6*t7,  50: (t6*t7)^-1,  51: j*k,  52: k*j,  53: t8^(j*k)
gap> Append( slp, [ [ 34, 1, 35, 1 ], [ 49, -1 ], [ 45, 1, 48, 1 ],
>                   [ 48, 1, 45, 1 ], [ 51, -1, 36, 1, 51, 1 ] ] );
gap> # 54: Comm( t8^(j*k), t6*t7),  55: t8^(k*j)
gap> Append( slp, [ [ 53, -1, 50, 1, 53, 1, 49, 1 ] ] );
gap> Append( slp, [ [ 52, -1, 36, 1, 52, 1 ] ] );
gap> # 56: Comm( t8^(k*j), t6*t7 )
gap> Append( slp, [ [ 55, -1, 50, 1, 55, 1, 49, 1 ] ] );
gap> # 57: l = Comm( t8^(j*k), t6*t7 ) * Comm( t8^(k*j), t6*t7 )
gap> Append( slp, [ [ 54, 1, 56, 1 ] ] );

##
gap> # 58: (j*k)^3,  59: (j*k)^-3,  60: t8^((j*k)^4)
gap> Append( slp, [ [ 51, 3 ], [ 58, -1 ], [ 59, 1, 53, 1, 58, 1 ] ] );
gap> # 61: l3 = Comm( t8^((j*k)^4), t6*t7 )
gap> Append( slp, [ [ 60, -1, 50, 1, 60, 1, 49, 1 ] ] );
gap> # 62: l4 = t8^((j*k)^3*k*j)
gap> Append( slp, [ [ 52, -1, 59, 1, 36, 1, 58, 1, 52, 1 ] ] );
gap> # 63: l3*l4,  64: l*l3*l4
gap> Append( slp, [ [ 61, 1, 62, 1 ], [ 57, 1, 63, 1 ] ] );
gap> # 65: l5:= ( l * l3 * l4 )^3 * l3 * l4;;
gap> Append( slp, [ [ 64, 3, 63, 1 ] ] );

##
gap> # 66: l5^4,  67: m2 = l4^(l5^4)
gap> Append( slp, [ [ 65, 4 ], [ 66, -1, 62, 1, 66, 1 ] ] );
gap> # 68: m3 = m2^l5
gap> Append( slp, [ [ 65, -1, 67, 1, 65, 1 ] ] );
gap> # 69: t10 = m3*m2*l4*m2*m3
gap> Append( slp, [ [ 68, 1, 67, 1, 62, 1, 67, 1, 68, 1 ] ] );
gap> resultpos[10]:= Length( slp ) + 2;;
gap> # 70: t9 = l4*m2*t10*m2*l4
gap> Append( slp, [ [ 62, 1, 67, 1, 69, 1, 67, 1, 62, 1 ] ] );
gap> resultpos[9]:= Length( slp ) + 2;;

##
gap> Add( slp, List( resultpos, x -> [ x, 1 ] ) );
gap> slp:= StraightLineProgram( slp, 2 );
<straight line program>

##
gap> b_2:= AtlasGroup( "B", Characteristic, 2, Dimension, 4370 );;
gap> b_3:= AtlasGroup( "B", Characteristic, 3, Dimension, 4371 );;
gap> b_5:= AtlasGroup( "B", Characteristic, 5, Dimension, 4371 );;
gap> gens_2:= GeneratorsOfGroup( b_2 );;
gap> gens_3:= GeneratorsOfGroup( b_3 );;
gap> gens_5:= GeneratorsOfGroup( b_5 );;
gap> res_2:= ResultOfStraightLineProgram( slp, gens_2 );;
gap> ForAll( relsslps,
>            prg -> IsOne( ResultOfStraightLineProgram( prg, res_2 ) ) );
true
gap> res_3:= ResultOfStraightLineProgram( slp, gens_3 );;
gap> ForAll( relsslps,
>            prg -> IsOne( ResultOfStraightLineProgram( prg, res_3 ) ) );
true
gap> res_5:= ResultOfStraightLineProgram( slp, gens_5 );;
gap> ForAll( relsslps,
>            prg -> IsOne( ResultOfStraightLineProgram( prg, res_5 ) ) );
true

##
gap> revslp:= [ Concatenation( List( [ 1 .. 8 ], i -> [ i, 1 ] ) ),
>        Concatenation( List( [ 5, 9, 10, 11 ], i -> [ i, 1 ] ) ) ];;

##
gap> Append( revslp, [ [ 12, 7, 13, 1 ], [ 14, 15 ], [ 13, 1, 12, 1 ],
>                      [ 16, 10 ], [ 17, -1 ], [ 1, 1, 2, 1 ],
>                      [ 18, 1, 19, 1, 17, 1 ] ] );

##
gap> Add( revslp, List( [ 15, 20 ], x -> [ x, 1 ] ) );
gap> revslp:= StraightLineProgram( revslp, 11 );
<straight line program>

##
gap> a:= gens_2[1];; b:= gens_2[2];;
gap> w:= One( a ) + b*a + a*b + b;;
gap> nsp_2:= NullspaceMat( w );; Length( nsp_2 );
1
gap> a:= gens_3[1];; b:= gens_3[2];;
gap> w:= One( a ) + b*a + a*b + b;;
gap> nsp_3:= NullspaceMat( w );; Length( nsp_3 );
1
gap> a:= gens_5[1];; b:= gens_5[2];;
gap> w:= One( a ) + b*a + a*b + b;;
gap> nsp_5:= NullspaceMat( w );; Length( nsp_5 );
1

##
gap> StdBasis:= function( F, mats, seed )
>      local n, b, mb, v, m, new;

>      n:= Length( mats[1] );
>      b:= [ seed ];
>      mb:= MutableBasis( F, b );
>      for v in b do
>        for m in mats do
>          new:= v * m;
>          if not IsContainedInSpan( mb, new ) then
>            Add( b, new );
>            if Length( b ) = n then
>              break;
>            fi;
>            CloseMutableBasis( mb, new );
>          fi;
>        od;
>        if Length( b ) = n then
>          break;
>        fi;
>      od;
>      return b;
>    end;;

##
gap> stdbas_2:= StdBasis( GF(2), gens_2, nsp_2[1] );;
gap> inv:= stdbas_2^-1;;
gap> stdgens_2:= List( gens_2, m -> stdbas_2 * m * inv );;
gap> newgens_2:= ResultOfStraightLineProgram( revslp, res_2 );;
gap> aa:= newgens_2[1];;  bb:= newgens_2[2];;
gap> neww:= One( aa ) + bb * aa + aa * bb + bb;;
gap> newnsp_2:= NullspaceMat( neww );;  Length( newnsp_2 );
1
gap> newstdbas_2:= StdBasis( GF(2), newgens_2, newnsp_2[1] );;
gap> inv:= newstdbas_2^-1;;
gap> newstdgens_2:= List( newgens_2, m -> newstdbas_2 * m * inv );;
gap> stdgens_2 = newstdgens_2;
true

##
gap> stdbas_3:= StdBasis( GF(3), gens_3, nsp_3[1] );;
gap> inv:= stdbas_3^-1;;
gap> stdgens_3:= List( gens_3, m -> stdbas_3 * m * inv );;
gap> newgens_3:= ResultOfStraightLineProgram( revslp, res_3 );;
gap> aa:= newgens_3[1];;  bb:= newgens_3[2];;
gap> neww:= One( aa ) + bb * aa + aa * bb + bb;;
gap> newnsp_3:= NullspaceMat( neww );;  Length( newnsp_3 );
1
gap> newstdbas_3:= StdBasis( GF(3), newgens_3, newnsp_3[1] );;
gap> inv:= newstdbas_3^-1;;
gap> newstdgens_3:= List( newgens_3, m -> newstdbas_3 * m * inv );;
gap> stdgens_3 = newstdgens_3;
true

##
gap> stdbas_5:= StdBasis( GF(5), gens_5, nsp_5[1] );;
gap> inv:= stdbas_5^-1;;
gap> stdgens_5:= List( gens_5, m -> stdbas_5 * m * inv );;
gap> newgens_5:= ResultOfStraightLineProgram( revslp, res_5 );;
gap> aa:= newgens_5[1];;  bb:= newgens_5[2];;
gap> neww:= One( aa ) + bb * aa + aa * bb + bb;;
gap> newnsp_5:= NullspaceMat( neww );;  Length( newnsp_5 );
1
gap> newstdbas_5:= StdBasis( GF(5), newgens_5, newnsp_5[1] );;
gap> inv:= newstdbas_5^-1;;
gap> newstdgens_5:= List( newgens_5, m -> newstdbas_5 * m * inv );;
gap> stdgens_5 = newstdgens_5;
true

##
gap> IdentifyClassName:= function( data2, data3, data5, slp, order )
>      local data, mats, elm, cand, nams, trace, pos, one, rank;

>      data:= [ , data2, data3,, data5 ];
>      mats:= [];

>      elm:= function( p )
>        if not IsBound( mats[p] ) then
>          if slp = fail then
>            mats[p]:= data[p];
>          else
>            mats[p]:= ResultOfStraightLineProgram( slp, data[p] );
>          fi;
>        fi;
>        return mats[p];
>      end;

>      if order = fail then
>        order:= Order( elm(2) );
>      fi;

>      # The element order suffices in certain cases.
>      if order in [ 23, 31, 46, 47, 56 ] then
>        # There are two Galois conjugate classes of elements of this order.
>        return Concatenation( String( order ), "AB" );
>      elif order in [ 1, 7, 11, 13, 17, 19, 21, 25, 27, 33, 35, 38, 39,
>                      44, 47, 52, 55, 66, 70 ] then
>        # There is exactly one conjugacy class of elements of this order.
>        return Concatenation( String( order ), "A" );
>      fi;

>      if order in [ 3, 5, 9, 15 ] then
>        # The trace in the 2-modular representation suffices.
>        cand:= [ [3,1], [3,0], [5,0], [5,1], [9,0], [9,1], [15,0], [15,1] ];
>        nams:= [ "3A", "3B", "5A", "5B", "9A", "9B", "15A", "15B" ];
>        trace:= Int( TraceMat( elm(2) ) );
>        return nams[ Position( cand, [ order, trace ] ) ];
>      elif order mod 4 = 2 then
>        # Compute the rank of 1 + x.
>        cand:= [ [ 2, 1860 ], [ 2, 2048 ], [ 2, 2158 ], [ 2, 2168 ],
>                 [ 6, 3486 ], [ 6, 3510 ], [ 6, 3566 ], [ 6, 3534 ],
>                 [ 6, 3606 ], [ 6, 3604 ], [ 6, 3596 ], [ 6, 3610 ],
>                 [ 6, 3636 ], [ 6, 3638 ], [ 6, 3634 ],
>                 [ 10, 3860 ], [ 10, 3896 ], [ 10, 3918 ], [ 10, 3908 ],
>                 [ 10, 3920 ], [ 10, 3932 ],
>                 [ 14, 3996 ], [ 14, 4008 ], [ 14, 4048 ], [ 14,4034 ],
>                 [ 14, 4052 ],
>                 [ 18, 4088 ], [ 18, 4090 ], [ 18, 4110 ], [ 18, 4124 ],
>                 [ 18, 4128 ], [ 18, 4122 ],
>                 [ 22, 4140 ], [ 22, 4158 ],
>                 [ 26, 4198 ], [ 26, 4176 ],
>                 [ 30, 4190 ], [ 30, 4212 ], [ 30, 4206 ], [ 30, 4214 ],
>                 [ 30, 4224 ], [ 30, 4216 ],
>                 [ 34, 4238 ], [ 34, 4220 ],
>                 [ 42, 4242 ], [ 42, 4258 ] ];
>        nams:= [ "2A", "2B", "2C", "2D",
>                 "6A", "6B", "6C", "6D", "6E", "6F",
>                 "6G", "6H", "6I", "6J", "6K",
>                 "10A", "10B", "10C", "10D", "10E", "10F",
>                 "14A", "14B", "14C", "14D", "14E",
>                 "18A", "18B", "18C", "18D", "18E", "18F",
>                 "22A", "22B",
>                 "26A", "26B",
>                 "30AB", "30C", "30D", "30E", "30F", "30GH",
>                 "34A", "34BC",
>                 "42AB", "42C" ];
>        one:= elm(2)^0;
>        rank:= RankMat( elm(2) + one );
>        pos:= Position( cand, [ order, rank ] );
>        if nams[ pos ] = "30AB" then
>          rank:= RankMat( elm(2)^5 + one );
>          if   rank = 3510 then return "30A";
>          elif rank = 3486 then return "30B";
>          else Error( "wrong rank" );
>          fi;
>        elif nams[ pos ] = "42AB" then
>          rank:= RankMat( elm(2)^3 + one );
>          if   rank = 3996 then return "42A";
>          elif rank = 4008 then return "42B";
>          else Error( "wrong rank" );
>          fi;
>        else
>          return nams[ pos ];
>        fi;
>      elif order in [ 36, 60 ] then
>        cand:= [ [36,4226],[36,4238],[36,4248],[60,4280],[60,4286],[60,4296] ];
>        nams:= [ "36A", "36B", "36C", "60A", "60B", "60C" ];
>        one:= elm(2)^0;
>        rank:= RankMat( elm(2) + one );
>        return nams[ Position( cand, [ order, rank ] ) ];
>      elif order = 28 then
>        one:= elm(2)^0;
>        rank:= RankMat( elm(2) + one );
>        trace:= Int( TraceMat( elm(3)^7 ) );
>        if rank = 4188 then   # 28A or 28C
>          if   trace = 0 then return "28A";
>          elif trace = 1 then return "28C";
>          else Error( "wrong trace" );
>          fi;
>        elif rank = 4200 then   # 28B or 28D
>          if   trace = 1 then return "28B";
>          elif trace = 0 then return "28D";
>          else Error( "wrong trace" );
>          fi;
>        elif rank = 4210 then return "28E";
>        else Error( "wrong rank" );
>        fi;
>      elif order = 32 then
>        trace:= Int( TraceMat( elm(3)^2 ) );
>        if   trace = 2 then return "32AB";
>        elif trace = 0 then return "32CD";
>        else Error( "wrong trace" );
>        fi;
>      elif order = 40 then
>        one:= elm(2)^0;
>        rank:= RankMat( elm(2) + one );
>        if rank = 4242 then   # 40A or 40B 0r 40C
>          trace:= Int( TraceMat( elm(3) ) );
>          if   trace = 0 then return "40A";
>          elif trace = 1 then return "40B";
>          else return "40C";
>          fi;
>        elif rank = 4250 then return "40D";
>        elif rank = 4258 then return "40E";
>        else Error( "wrong rank" );
>        fi;
>      elif order = 48 then
>        trace:= Int( TraceMat( elm(3) ) );
>        if   trace = 0 then return "48A";
>        elif trace = 1 then return "48B";
>        else Error( "wrong trace" );
>        fi;
>      elif order in [ 4, 8 ] then
>        cand:= [ [4,3114],[4,3192],[4,3256],[4,3202],[4,3204],[4,3266],
>                 [4,3264],[8,3774],[8,3738],[8,3778],[8,3780],[8,3810],
>                 [8,3786],[8,3812],[8,3818] ];
>        nams:= [ "4A-B", "4C-D", "4E", "4F", "4G", "4H-J", "4I",
>                 "8A", "8B-C-E", "8D", "8F-H", "8G", "8I-L", "8J", "8K-M-N" ];
>        one:= elm(2)^0;
>        rank:= RankMat( elm(2) + one );
>        pos:= Position( cand, [ order, rank ] );
>        if not '-' in nams[ pos ] then
>          return nams[ pos ];
>        elif order = 4 then
>          trace:= Int( TraceMat( elm(3) ) );
>          if trace = 0 then
>            if   nams[ pos ] = "4A-B" then return "4B";
>            elif nams[ pos ] = "4C-D" then return "4D";
>            else return "4H";
>            fi;
>          elif trace = 1 then
>            if   nams[ pos ] = "4A-B" then return "4A";
>            elif nams[ pos ] = "4C-D" then return "4C";
>            else return "4J";
>            fi;
>          else
>            Error( "wrong trace" );
>          fi;
>        elif order = 8 then
>          if nams[ pos ] = "8B-C-E" then
>            trace:= Int( TraceMat( elm(3) ) );
>            if   trace = 1 then return "8B";
>            elif trace = 0 then return "8C";
>            else return "8E";
>            fi;
>          elif nams[ pos ] = "8F-H" then
>            rank:= RankMat( ( elm(2) + one )^3 );
>            if   rank = 2619 then return "8F";
>            elif rank = 2620 then return "8H";
>            else Error( "wrong rank" );
>            fi;
>          elif nams[ pos ] = "8I-L" then
>            rank:= RankMat( ( elm(2) + one )^2 );
>            if   rank = 3202 then return "8I";
>            elif rank = 3204 then return "8L";
>            else Error( "wrong rank" );
>            fi;
>          else   # 8K-M-N
>            rank:= RankMat( ( elm(2) + one )^3 );
>            if rank = 2714 then   # 8K-M
>              trace:= Int( TraceMat( elm(3) ) );
>              if   trace = 1 then return "8K";
>              elif trace = 2 then return "8M";
>              else Error( "wrong trace" );
>              fi;
>            elif rank = 2717 then return "8N";
>            else Error( "wrong rank" );
>            fi;
>          fi;
>        fi;
>      elif order in [ 12, 24 ] then
>        cand:= [ [12,3936],[12,3942],[12,3958],[12,3996],[12,3962],[12,3964],
>                 [12,3986],[12,3978],[12,3966],[12,4000],[12,3982],[12,3988],
>                 [12,4002],[12,4004],[24,4152],[24,4164],[24,4170],[24,4182],
>                 [24,4176],[24,4178],[24,4174],[24,4186] ];
>        nams:= [ "12A-C-D", "12B", "12E", "12F", "12G-H", "12I", "12J",
>                 "12K-M", "12L", "12N", "12O", "12P", "12Q-R-T", "12S",
>                 "24A-B-C-D", "24E-G", "24F", "24H", "24I-M", "24J", "24K",
>                 "24L-N" ];
>        one:= elm(2)^0;
>        rank:= RankMat( elm(2) + one );
>        pos:= Position( cand, [ order, rank ] );
>        if not '-' in nams[ pos ] then
>          return nams[ pos ];
>        elif order = 12 then
>          if nams[ pos ] = "12A-C-D" then
>            trace:= Int( TraceMat( elm(5) ) );
>            if   trace = 3 then return "12A";
>            elif trace = 4 then return "12C";
>            elif trace = 1 then return "12D";
>            else Error( "wrong trace" );
>            fi;
>          else
>            trace:= Int( TraceMat( elm(3) ) );
>            if trace = 0 then
>              if   nams[ pos ] = "12G-H" then return "12H";
>              elif nams[ pos ] = "12K-M" then return "12K";
>              else return "12Q";  # 12Q-R-T
>              fi;
>            elif trace = 1 then
>              if   nams[ pos ] = "12G-H" then return "12G";
>              elif nams[ pos ] = "12K-M" then return "12M";
>              else return "12R";  # 12Q-R-T
>              fi;
>            elif nams[ pos ] = "12Q-R-T" then
>              return "12T";
>            else
>              Error( "wrong trace" );
>            fi;
>          fi;
>        elif order = 24 then
>          if nams[ pos ] = "24I-M" then
>            rank:= RankMat( elm(2)^2 + one );
>            if   rank = 3986 then return "24I";
>            elif rank = 3982 then return "24M";
>            else Error( "wrong rank" );
>            fi;
>          elif nams[ pos ] = "24E-G" then
>            rank:= RankMat( elm(2)^3 + one );
>            if   rank = 3774 then return "24E";
>            elif rank = 3778 then return "24G";
>            else Error( "wrong rank" );
>            fi;
>          elif nams[ pos ] = "24L-N" then
>            trace:= Int( TraceMat( elm(3) ) );
>            if   trace = 1 then return "24L";
>            elif trace = 2 then return "24N";
>            else Error( "wrong trace" );
>            fi;
>          else   # 24A-B-C-D"
>            trace:= Int( TraceMat( elm(5) ) );
>            if   trace = 3 then return "24A";
>            elif trace = 0 then return "24B";
>            elif trace = 2 then return "24C";
>            elif trace = 1 then return "24D";
>            else Error( "wrong trace" );
>            fi;
>          fi;
>        fi;
>      elif order in [ 16, 20 ] then
>        cand:= [ [ 16, 4072 ], [ 16, 4074 ], [ 16, 4094 ],
>                 [ 20, 4114 ], [ 20, 4128 ], [ 20, 4132 ], [ 20, 4148 ],
>                 [ 20, 4144 ], [ 20, 4138 ], [ 20, 4150 ] ];
>        nams:= [ "16A-B", "16C-D-E-F", "16G-H",
>                 "20A-B-C-D", "20E", "20F", "20G", "20H", "20I", "20J" ];
>        one:= elm(2)^0;
>        rank:= RankMat( elm(2) + one );
>        pos:= Position( cand, [ order, rank ] );
>        if not '-' in nams[ pos ] then
>          return nams[ pos ];
>        elif order = 20 then
>          rank:= RankMat( elm(2)^2 + one );
>          if   rank = 3908 then return "20B";
>          elif rank <> 3896 then Error( "wrong rank" );
>          else
>            trace:= Int( TraceMat( elm(3) ) );
>            if   trace = 2 then return "20A";
>            elif trace = 0 then return "20C";
>            elif trace = 1 then return "20D";
>            else Error( "wrong trace" );
>            fi;
>          fi;
>        else   # order = 16
>          if nams[ pos ] = "16A-B" then
>            trace:= Int( TraceMat( elm(3) ) );
>            if   trace = 0 then return "16A";
>            elif trace = 1 then return "16B";
>            else Error( "wrong trace" );
>            fi;
>          elif nams[ pos ] = "16G-H" then
>            trace:= Int( TraceMat( elm(3)^2 ) );
>            if   trace = 1 then return "16G";
>            elif trace = 2 then return "16H";
>            else Error( "wrong trace" );
>            fi;
>          else   # 16C-D-E-F
>            trace:= Int( TraceMat( elm(3) ) );
>            if trace = 0 then   # We cannot distinguish 16D and 16F.
>              return "16DF";
>            elif trace = 2 then   # 16C-E
>              one:= elm(2)^0;
>              rank:= RankMat( elm(2)^2 + one );
>              if   rank = 3780 then return "16C";
>              elif rank = 3778 then return "16E";
>              else Error( "wrong rank" );
>              fi;
>            else
>              Error( "wrong trace" );
>            fi;
>          fi;
>        fi;
>      else Error( "wrong element order" );
>      fi;
>  end;;

##
gap> cycprg:= AtlasProgram( "B", "cyclic" );;
gap> cycprg.identifier;
[ "B", "BG1-cycW1", 1 ]
gap> cycprg.outputs;
[ "12A", "12H", "12I", "12L", "12P", "12S", "12T", "16E", "16F", "16G", 
  "16H", "18A", "18B", "18D", "18F", "20B", "20C", "20H", "20I", "20J", 
  "24A", "24B", "24C", "24D", "24E", "24F", "24H", "24I", "24J", "24K", 
  "24L", "24M", "24N", "25A", "26B", "27A", "28A", "28C", "28D", "28E", 
  "30A", "30B", "30C", "30E", "30G-H", "31A-B", "32A-B", "32C-D", "34A", 
  "34BC", "36A", "36B", "36C", "38A", "39A", "40A", "40B", "40C", "40D", 
  "40E", "42A", "42B", "42C", "44A", "46AB", "47AB", "48A", "48B", "52A", 
  "55A", "56AB", "60A", "60B", "60C", "66A", "70A" ]

##
gap> DefinitionsViaPowerMaps:= [
>     [ "70A", 2, "35A" ], [ "66A", 2, "33A" ], [ "60A", 2, "30D" ],
>     [ "60C", 2, "30F" ], [ "56AB", 2, "28B" ], [ "52A", 2, "26A" ],
>     [ "48B", 2, "24G" ], [ "46AB", 2, "23AB" ], [ "44A", 2, "22B" ],
>     [ "66A", 3, "22A" ], [ "42C", 2, "21A" ], [ "40E", 2, "20G" ],
>     [ "40D", 2, "20F" ], [ "60B", 3, "20E" ], [ "60A", 3, "20A" ],
>     [ "40C", 2, "20D" ], [ "38A", 2, "19A" ], [ "36C", 2, "18E" ],
>     [ "36B", 2, "18C" ], [ "34A", 2, "17A" ], [ "32CD", 2, "16DF" ],
>     [ "32AB", 2, "16C" ], [ "48B", 3, "16B" ], [ "48A", 3, "16A" ],
>     [ "30F", 2, "15B" ], [ "30A", 2, "15A" ], [ "28E", 2, "14E" ],
>     [ "28A", 2, "14D" ], [ "42A", 3, "14A" ], [ "42B", 3, "14B" ],
>     [ "42C", 3, "14C" ], [ "26A", 2, "13A" ], [ "24N", 2, "12R" ],
>     [ "24M", 2, "12O" ], [ "24L", 2, "12Q" ], [ "24K", 2, "12M" ],
>     [ "24J", 2, "12J" ], [ "24H", 2, "12F" ], [ "24G", 2, "12G" ],
>     [ "24D", 2, "12D" ], [ "36C", 3, "12N" ], [ "36B", 3, "12K" ],
>     [ "36A", 3, "12B" ], [ "60A", 5, "12C" ], [ "60B", 5, "12E" ],
>     [ "22B", 2, "11A" ], [ "20J", 2, "10F" ], [ "20I", 2, "10D" ],
>     [ "20H", 2, "10C" ], [ "20F", 2, "10B" ], [ "30A", 3, "10A" ],
>     [ "30E", 3, "10E" ], [ "18F", 2, "9B" ], [ "18E", 2, "9A" ],
>     [ "16H", 2, "8M" ], [ "16G", 2, "8K" ], [ "16DF", 2, "8H" ],
>     [ "16E", 2, "8D" ], [ "24J", 3, "8J" ], [ "24M", 3, "8I" ],
>     [ "24I", 3, "8G" ], [ "24K", 3, "8F" ], [ "24C", 3, "8E" ],
>     [ "24B", 3, "8C" ], [ "24A", 3, "8B" ], [ "24E", 3, "8A" ],
>     [ "24N", 3, "8N" ], [ "40D", 5, "8L" ], [ "14D", 2, "7A" ],
>     [ "12T", 2, "6K" ], [ "12S", 2, "6J" ], [ "12R", 2, "6I" ],
>     [ "12P", 2, "6H" ], [ "12O", 2, "6G" ], [ "12I", 2, "6C" ],
>     [ "18A", 3, "6D" ], [ "30B", 5, "6A" ], [ "30A", 5, "6B" ],
>     [ "30E", 5, "6E" ], [ "30C", 5, "6F" ], [ "12C", 3, "4A" ],
>     [ "10F", 2, "5B" ], [ "10B", 2, "5A" ], [ "8N", 2, "4J" ],
>     [ "8M", 2, "4H" ], [ "8L", 2, "4G" ], [ "8J", 2, "4E" ],
>     [ "8I", 2, "4F" ], [ "8H", 2, "4C" ], [ "8E", 2, "4B" ],
>     [ "12E", 3, "4D" ], [ "12T", 3, "4I" ], [ "6K", 2, "3B" ],
>     [ "6A", 2, "3A" ], [ "4J", 2, "2D" ], [ "4I", 2, "2C" ],
>     [ "4A", 2, "2B" ], [ "6A", 3, "2A" ], [ "2B", 2, "1A" ],
>    ];;

##
gap> SLPForClassName:= function( nam, cycslp, outputnames )
>      local pos, rule;

>      pos:= Position( outputnames, nam );
>      if pos <> fail then
>        return RestrictOutputsOfSLP( cycslp.program, pos );
>      fi;

>      rule:= First( DefinitionsViaPowerMaps, x -> x[3] = nam );
>      if rule = fail then
>        Error( "'nam' is not an admiss. name for a cyclic subgroup of B" );
>      fi;

>      return CompositionOfStraightLinePrograms(
>                 StraightLineProgram( [ [ 1, rule[2] ] ], 1 ),
>                 SLPForClassName( rule[1], cycslp, outputnames ) );
> end;;

##
gap> outputnames:= List( cycprg.outputs,
>                        x -> ReplacedString( x, "-", "" ) );;
gap> outputnames:= List( outputnames,
>                        x -> ReplacedString( x, "16F", "16DF" ) );;
gap> labels:= Union( outputnames,
>                    List( DefinitionsViaPowerMaps, x -> x[3] ) );;
gap> for l in labels do
>      slp:= SLPForClassName( l, cycprg, outputnames );
>      id:= IdentifyClassName( gens_2, gens_3, gens_5, slp, fail );
>      if id <> l then
>        Print( "#E  problem with identification: ", id, " vs. ", l, "\n" );
>      fi;
>    od;

##
gap> SortParallel( List( labels, x -> Int( Filtered( x, IsDigitChar ) ) ),
>                  labels );
gap> powerinfo:= [];;
gap> for l in labels do
>      slp:= SLPForClassName( l, cycprg, outputnames );
>      ord:= Int( Filtered( l, IsDigitChar ) );
>      pow:= [];
>      if not ( IsPrimeInt( ord ) or ord = 1 ) then
>        for p in Set( Factors( ord ) ) do
>          powerslp:= CompositionOfStraightLinePrograms(
>                         StraightLineProgram( [ [ 1, p ] ], 1 ), slp );
>          id:= IdentifyClassName( gens_2, gens_3, gens_5, powerslp,
>                                  ord / p );
>          Add( pow, [ p, id ] );
>        od;
>      fi;
>      Add( powerinfo, [ l, pow ] );
>    od;

##
gap> h:= CharacterTable( "2.2E6(2).2" );;
gap> invpos:= Positions( OrdersClassRepresentatives( h ), 2 );
[ 2, 3, 4, 5, 6, 7, 175, 176, 177, 178 ]
gap> ClassNames( h ){ invpos };
[ "2a", "2b", "2c", "2d", "2e", "2f", "2g", "2h", "2i", "2j" ]
gap> AtlasClassNames( h ){ invpos };
[ "1A_1", "2A_0", "2A_1", "2B_0", "2B_1", "2C_0", "2D_0", "2D_1", "2E_0", 
  "2E_1" ]

##
gap> f:= CharacterTable( "2E6(2).2" );;
gap> index1:= 3968055;;
gap> constit:= Filtered( Irr( f ), x -> x[1] <= index1 );;
gap> degrees:= Set( constit, x -> x[1] );
[ 1, 1938, 48620, 554268, 815100, 1322685, 1828332, 2089164, 2909907, 2956096 
 ]
gap> lincom:= Filtered( UnorderedTuples( degrees, 5 ),
>                       x -> Sum( x ) = index1 );
[ [ 1, 1938, 48620, 1828332, 2089164 ] ]

##
gap> degrees:= lincom[1];
[ 1, 1938, 48620, 1828332, 2089164 ]
gap> constit:= List( degrees, d -> Filtered( constit, x -> x[1] = d ) );;
gap> cand1:= List( Cartesian( constit ), Sum );;
gap> cand1:= Filtered( cand1, x -> ForAll( x, y -> y >= 0 ) );;
gap> List( ConstituentsOfCharacter( cand1[1] ),
>          x -> Position( Irr( f ), x ) );
[ 1, 3, 5, 13, 15 ]

##
gap> PermComb( f, rec( degree:= index1 ) ) = cand1;
true

##
gap> u:= CharacterTable( "F4(2)" );;
gap> ufush:= PossibleClassFusions( u, h );;
gap> ind:= Set( ufush,
>               map -> InducedClassFunctionsByFusionMap( u, h,
>                          [ TrivialCharacter( u ) ], map )[1] );;
gap> Length( ind );
1
gap> const:= ConstituentsOfCharacter( ind[1] );;
gap> Sum( const ) = ind[1];
true
gap> sub:= List( Cartesian( List( const, x -> [ Zero( x ), x ] ) ), Sum );;
gap> cand2:= Filtered( sub,
>                      x -> x[1] = ind[1][1] / 4 and Minimum( x ) >= 0 );;
gap> Length( cand2 );
1
gap> List( ConstituentsOfCharacter( cand2[1] ),
>          x -> Position( Irr( h ), x ) );
[ 1, 5, 17, 24 ]

##
gap> s:= CharacterTable( "Fi22.2" );;
gap> sfush:= PossibleClassFusions( s, h );;
gap> Length( sfush );
2
gap> Positions( OrdersClassRepresentatives( s ), 2 );
[ 2, 3, 4, 60, 61, 62 ]
gap> List( sfush, x -> x[60] );
[ 175, 176 ]
gap> cand3:= InducedClassFunctionsByFusionMap( s, h,
>                [ TrivialCharacter( s ) ], sfush[1] );;
gap> SortedList( List( ConstituentsOfCharacter( cand3[1] ),
>                      x -> Position( Irr( h ), x ) ) );
[ 1, 3, 5, 13, 17, 28, 49, 76, 190, 192, 196, 202, 210, 217 ]

##
gap> h:= CharacterTable( "2.2E6(2).2" );;
gap> u:= CharacterTable( "U4(3)" );;
gap> Positions( OrdersClassRepresentatives( h ), 3 );
[ 8, 10, 12 ]
gap> ufush:= PossibleClassFusions( u, h );;
gap> 3pos:= Positions( OrdersClassRepresentatives( u ), 3 );
[ 3, 4, 5, 6 ]
gap> Set( ufush, x -> x{ 3pos } );
[ [ 12, 8, 10, 10 ], [ 12, 10, 8, 10 ] ]
gap> u:= CharacterTable( "2.U4(3)" );;
gap> ufush:= PossibleClassFusions( u, h );;
gap> 3pos:= Positions( OrdersClassRepresentatives( u ), 3 );
[ 5, 7, 9, 11 ]
gap> Set( ufush, x -> x{ 3pos } );
[ [ 12, 8, 10, 10 ], [ 12, 10, 8, 10 ] ]

##
gap> u:= CharacterTable( "U4(3)" ) mod 2;;
gap> phi:= Filtered( Irr( u ), x -> x[1] = 20 );;
gap> Display( u, rec( chars:= phi, powermap:= false ) );
U4(3)mod2

     2  7  3  2  2  .  .  .  .  .  .  .  .
     3  6  6  5  5  4  .  .  .  3  3  3  3
     5  1  .  .  .  .  1  .  .  .  .  .  .
     7  1  .  .  .  .  .  1  1  .  .  .  .

       1a 3a 3b 3c 3d 5a 7a 7b 9a 9b 9c 9d

Y.1    20 -7  2  2  2  . -1 -1 -1 -1 -1 -1

##
gap> 3pos:= Positions( OrdersClassRepresentatives( f ), 3 );;
gap> val3C:= 1 + cand1[1][ 3pos[3] ];;
gap> 3pos:= Positions( OrdersClassRepresentatives( h ), 3 );;
gap> val3C:= val3C + cand2[1][ 3pos[3] ] + cand3[1][ 3pos[3] ];;
gap> val3C:= val3C + SizesCentralizers( h )[ 3pos[3] ] / ( 2^8 * 3^6 );
1620

##
gap> Collected( Factors( val3C * SizesCentralizers( h )[ 3pos[3] ] ) );
[ [ 2, 13 ], [ 3, 13 ], [ 5, 1 ] ]

##
gap> slp_maxes_2:= AtlasProgram( "B", "maxes", 2 );;
gap> cgens_3:= ResultOfStraightLineProgram( slp_maxes_2.program,
>                  GeneratorsOfGroup( b_3 ) );;

##
gap> m:= GModuleByMats( cgens_3, GF(3) );;
gap> cf_3:= MTX.CompositionFactors( m );;
gap> SortParallel( List( cf_3, x -> x.dimension ), cf_3 );
gap> List( cf_3, x -> x.dimension );
[ 23, 2048, 2300 ]

##
gap> slp_co2:= StraightLineProgram( [
>     [ 2, 1, 1, 1 ], [ 2, 2 ], [ 4, 1, 1, 1 ], [ 4, 2 ],
>     [ 6, 1, 1, 1 ], [ 7, 4 ], [ 3, 2 ], [ 5, 1, 9, 1 ],
>     [ 10, -1 ], [ 6, 3 ], [ 11, 1, 8, 1 ], [ 13, 1, 10, 1 ],
>     [ [ 12, 1 ], [ 14, 1 ] ] ], 2 );;
gap> f:= FreeGroup( "x", "y" );;  x:= f.1;;  y:= f.2;;
gap> words:= ResultOfStraightLineProgram( slp_co2, [ x, y ] );;
gap> words = [ y^12, ((y^4*x)^4)^(y*(y*x)^3) ];
true
gap> co2gens:= cf_3[1].generators;;
gap> co2stdgens:= ResultOfStraightLineProgram( slp_co2, co2gens );;

##
gap> fgens:= cf_3[3].generators;;
gap> fstdgens:= ResultOfStraightLineProgram( slp_co2, fgens );;
gap> slp_co2m1:= AtlasProgram( "Co2", "maxes", 1 );;
gap> ugens:= ResultOfStraightLineProgram( slp_co2m1.program, fstdgens );;
gap> one:= ugens[1]^0;;
gap> comm:= Comm( ugens[1], ugens[2] );;
gap> Order( comm );
12
gap> pow:= comm^2;;
gap> mats:= List( [ pow, pow^ugens[1] ], x -> x - one );;
gap> nsp:= List( mats, NullspaceMat );;
gap> List( nsp, Length );
[ 434, 434 ]
gap> si:= SumIntersectionMat( nsp[1], nsp[2] );;
gap> Length( si[2] );
1
gap> orb:= Orbit( Group( fstdgens ), si[2][1] );;
gap> Length( orb );
4600
gap> orb:= SortedList( orb );;
gap> stdperms:= List( fstdgens, x -> Permutation( x, orb ) );;
gap> List( stdperms, Order );
[ 2, 5 ]

##
gap> slp_ker:= StraightLineProgram( [
>     [ 1, 1, 2, 1 ], [ 2, 1, 1, 1 ], [ 2, 2 ], [ 4, 1, 2, 1 ],
>     [ 2, 1, 4, 1 ], [ 3, 1, 2, 1 ], [ 3, 2 ], [ 4, 2 ], [ 6, 2 ],
>     [ 7, 2 ], [ 2, 1, 10, 1 ], [ 2, 1, 6, 1 ], [ 10, 1, 2, 1 ],
>     [ 5, 1, 3, 1 ], [ 4, 1, 5, 1 ], [ 9, 1, 1, 1 ], [ 3, 1, 10, 1 ],
>     [ 9, 1, 4, 1 ], [ 5, 1, 13, 1 ], [ 2, 1, 12, 1 ], [ 14, 1, 7, 1 ],
>     [ 17, 1, 7, 1 ], [ 5, 1, 15, 1 ], [ 12, 1, 2, 1 ], [ 6, 1, 14, 1 ],
>     [ 13, 1, 5, 1 ], [ 11, 1, 2, 1 ], [ 12, 1, 4, 1 ], [ 13, 1, 7, 1 ],
>     [ 11, 1, 4, 1 ], [ 11, 1, 3, 1 ], [ 12, 1, 10, 1 ],
>     [ [ 7, 9 ], [ 6, 9 ], [ 8, 9 ], [ 16, 9 ], [ 17, 9 ], [ 18, 9 ],
>       [ 19, 9 ], [ 20, 9 ], [ 21, 4 ], [ 22, 4 ], [ 23, 4 ], [ 24, 4 ],
>       [ 25, 4 ], [ 26, 4 ], [ 27, 4 ], [ 28, 4 ], [ 29, 4 ], [ 30, 12 ],
>       [ 31, 12 ], [ 32, 12 ], [ 33, 12 ], [ 34, 12 ] ] ], 2 );;
gap> f:= FreeGroup( "a", "b" );;  a:= f.1;;  b:= f.2;;
gap> ResultOfStraightLineProgram( slp_ker, [ a, b ] );
[ (b^2*a)^9, (b*a*b)^9, (a*b^2)^9, (b^2*a*b)^9, (b*a*b^2)^9, ((a*b)^2*a)^9, 
  (a*b*(b*a)^2)^9, ((a*b)^2*b*a)^9, (b^3*(b*a)^2)^4, (b*(b^2*a)^2)^4, 
  (b^2*a*b^3*a)^4, (b*a*b^4*a)^4, (b^2*(b*a)^2*b)^4, ((b^2*a)^2*b)^4, 
  (b*a*b^3*a*b)^4, (b*(b*a)^2*b^2)^4, ((b*a*b)^2*b)^4, ((b^2*a)^2*b*a)^12, 
  (b*(b*a)^2*b^2*a)^12, ((b*a*b)^2*b*a)^12, ((b*a*b)^2*a*b)^12, 
  ((b^2*a)^2*b*a*b*a)^12 ]

##
gap> slp_co2classreps:= AtlasProgram( "Co2", "classes" );;

##
gap> classrepsinfo:= [
>  [  "1A", [ [ 0 ], [ 3, 4, 22 ], [  ], [ 8 ], [ 2 ], [ 1 ] ] ], 
>  [  "2A", [ [ 1, 6, 8, 9, 11 ], [ 12 ], [ 1, 4, 9, 10, 19 ],
>             [ 5, 7 ], [ 1, 9 ], [ 1, 16 ], [  ], [ 1 ], [ 5 ],
>             [ 1, 5, 18 ], [ 3 ] ] ], 
>  [  "2B", [ [ 1, 3, 4, 17, 20 ], [ 2, 14, 17 ], [ 2, 10, 22 ],
>             [ 14, 21 ], [  ], [ 9 ], [ 1, 5 ], [ 10, 11 ], [ 6, 12 ],
>             [ 1 ], [ 2 ] ] ], 
>  [  "2C", [ [ 4, 11, 13, 20 ], [ 1, 10, 12, 17 ], [ 3, 18, 21 ],
>             [ 8, 13, 17 ], [ 16, 22 ], [ 4, 5, 15 ], [ 8, 13 ],
>             [ 1, 10 ], [ 1, 21 ], [ 3, 9, 18, 21 ], [ 8 ],
>             [ 1, 10, 12 ], [ 4 ], [ 21 ], [ 6 ], [ 1 ], [  ] ] ],
>  [  "3A", [ [  ], [ 21 ], [ 1 ] ] ], 
>  [  "3B", [ [ 17, 20 ], [ 17 ], [ 1, 5, 17 ], [ 3, 18 ], [ 1 ], [ 10 ],
>             [ 12 ], [ 3 ], [  ] ] ],
>  [  "4A", [ [ 11 ], [  ], [ 1 ], [ 1, 12 ], [ 2 ] ] ], 
>  [  "4B", [ [ 2, 7, 19, 20 ], [ 2, 12 ], [ 9, 21 ], [ 5 ], [  ],
>             [ 14 ], [ 9, 20 ], [ 2, 14 ], [ 6 ], [ 1 ], [ 8 ], [ 2 ] ] ], 
>  [  "4C", [ [ 3, 7, 11 ], [ 18, 22 ], [ 4, 6 ], [ 21 ], [ 2, 17, 18 ],
>             [ 1, 14 ], [ 5, 12, 17 ], [ 14 ], [ 3, 4 ], [ 3 ], [ 10 ],
>             [ 2 ], [ 1 ], [  ] ] ], 
>  [  "4D", [ [ 20 ], [ 9, 22 ], [ 8, 16 ], [ 6 ], [ 6, 17 ], [  ],
>             [ 1 ] ] ], 
>  [  "4E", [ [ 11, 19 ], [ 3, 7 ], [ 1, 22 ], [ 1, 2, 22 ], [ 1, 10 ],
>             [ 3, 18 ], [  ], [ 5, 20 ], [ 2 ], [ 1, 15 ], [ 1 ], [ 3 ],
>             [ 8 ] ] ], 
>  [  "4F", [ [ 3, 4 ], [ 3, 19, 21 ], [ 4, 16 ], [ 4, 13 ], [ 3, 7 ],
>             [ 4 ], [ 3 ], [ 4, 17 ], [ 13 ], [ 1, 7 ], [  ], [ 1, 9 ],
>             [ 4, 14 ], [ 17 ], [ 19 ], [ 1 ], [ 1, 4 ], [ 18 ],
>             [ 11 ] ] ], 
>  [  "4G", [ [ 10 ], [ 1, 4 ], [ 8 ], [ 20 ], [  ], [ 16 ], [ 1 ],
>             [ 2 ] ] ], 
>  [  "5A", [ [  ], [ 1 ] ] ], 
>  [  "5B", [ [  ], [ 1 ], [ 2 ], [ 19 ], [ 15, 19 ], [ 4 ], [ 8 ] ] ], 
>  [  "6A", [ [ 2 ], [  ], [ 1 ] ] ], 
>  [  "6B", [ [  ], [ 21 ], [ 1, 11 ], [ 1 ], [ 10 ] ] ], 
>  [  "6C", [ [ 8 ], [ 1, 14 ], [  ], [ 9 ], [ 1 ], [ 10 ], [ 2 ],
>             [ 3 ] ] ], 
>  [  "6D", [ [  ], [ 19 ], [ 12, 17 ], [ 1, 11 ], [ 7 ], [ 4, 10 ], [ 1 ], 
>             [ 2, 3 ], [ 1, 7 ], [ 10 ], [ 3 ] ] ], 
>  [  "6E", [ [ 9 ], [ 1, 8 ], [ 2 ], [ 22 ], [  ], [ 2, 3 ], [ 1 ],
>             [ 3, 4 ], [ 3 ], [ 13 ], [ 7 ], [ 6 ] ] ], 
>  [  "6F", [ [ 5, 10 ], [ 1, 2, 4 ], [ 4, 13 ], [ 10, 18 ], [ 4, 12 ],
>             [ 4, 9 ], [ 21 ], [ 4 ], [ 8 ], [ 1, 10 ], [ 1, 8, 18 ],
>             [ 10 ], [ 6 ], [ 1 ], [  ], [ 3 ] ] ], 
>  [  "7A", [ [ 1, 2 ], [ 2 ], [  ], [ 7 ], [ 1, 10 ], [ 1 ], [ 11 ] ] ], 
>  [  "8A", [ [ 2, 18 ], [  ], [ 2 ], [ 9 ], [ 7 ], [ 1 ] ] ], 
>  [  "8B", [ [ 8, 17 ], [ 1, 7 ], [ 11, 21 ], [ 2, 8 ], [  ], [ 1 ],
>             [ 2, 12 ], [ 7 ], [ 5 ], [ 2 ] ] ], 
>  [  "8C", [ [ 1, 8 ], [ 13 ], [ 9 ], [  ], [ 1 ] ] ], 
>  [  "8D", [ [ 1, 2 ], [ 7 ], [ 1 ], [ 6 ], [  ] ] ], 
>  [  "8E", [ [ 2, 12 ], [ 3, 22 ], [ 4, 22 ], [ 4, 12 ], [ 1, 11 ],
>             [ 2, 22 ], [ 6, 12 ], [ 9 ], [ 2, 10 ], [ 20 ], [ 1, 9 ],
>             [ 11 ], [ 10 ], [ 1 ], [  ], [ 7 ] ] ], 
>  [  "8F", [ [ 1, 10 ], [ 18 ], [ 4 ], [ 9 ], [ 20 ], [  ], [ 6 ],
>             [ 1 ] ] ], 
>  [  "9A", [ [  ], [ 4 ], [ 3 ], [ 5 ] ] ],
>  [ "10A", [ [  ], [ 1 ] ] ], 
>  [ "10B", [ [ 2 ], [ 12 ], [  ], [ 1 ], [ 9 ], [ 10 ], [ 14 ], [ 3 ] ] ], 
>  [ "10C", [ [ 1, 20 ], [ 5 ], [ 1, 8 ], [ 12 ], [ 2 ], [  ], [ 10 ],
>             [ 3 ], [ 1 ] ] ],
>  [ "11A", [ [ 2 ], [ 1 ], [ 3 ], [  ] ] ], 
>  [ "12A", [ [ 10 ], [  ], [ 1 ] ] ], 
>  [ "12B", [ [  ], [ 14 ], [ 1 ], [ 3 ], [ 4 ], [ 10 ] ] ], 
>  [ "12C", [ [ 19 ], [  ], [ 7 ], [ 1, 7 ], [ 1 ] ] ], 
>  [ "12D", [ [ 1 ], [ 11 ], [ 8 ], [ 15 ], [  ], [ 2 ] ] ], 
>  [ "12E", [ [ 3 ], [ 5 ], [  ] ] ], 
>  [ "12F", [ [ 9 ], [ 1, 15 ], [ 5 ], [  ], [ 2 ], [ 12 ], [ 8 ],
>             [ 1 ] ] ], 
>  [ "12G", [ [  ], [ 1 ], [ 3 ] ] ], 
>  [ "12H", [ [ 1, 4 ], [ 4, 14 ], [ 4, 9 ], [ 4 ], [ 14 ], [ 9 ], [ 1 ],
>             [  ], [ 11 ], [ 18 ] ] ], 
>  [ "14A", [ [ 1, 2 ], [ 1, 7 ], [ 10 ], [ 1 ], [ 19 ], [  ], [ 16 ] ] ], 
>  [ "14B", [ [ 7 ], [  ], [ 1 ], [ 11 ] ] ], 
>  [ "14C", [ [ 7 ], [  ], [ 1 ], [ 11 ] ] ], 
>  [ "15A", [ [  ], [ 2 ], [ 3 ], [ 1 ] ] ],
>  [ "15B", [ [  ] ] ], 
>  [ "15C", [ [  ] ] ],
>  [ "16A", [ [ 6 ], [ 11 ], [  ], [ 1 ] ] ], 
>  [ "16B", [ [ 3 ], [  ], [ 6 ], [ 1 ] ] ], 
>  [ "18A", [ [ 4 ], [  ], [ 5 ], [ 3 ] ] ], 
>  [ "20A", [ [ 6 ], [ 1 ], [  ], [ 7 ] ] ], 
>  [ "20B", [ [ 3 ], [ 1 ], [ 2 ], [  ] ] ],
>  [ "23A", [ [  ] ] ], 
>  [ "23B", [ [  ] ] ],
>  [ "24A", [ [ 18 ], [ 7 ], [  ], [ 1 ] ] ], 
>  [ "24B", [ [ 10 ], [  ], [ 1 ], [ 4 ] ] ], 
>  [ "28A", [ [ 5 ], [  ], [ 10 ], [ 1 ] ] ], 
>  [ "30A", [ [ 2 ], [  ], [ 1 ], [ 3 ] ] ],
>  [ "30B", [ [  ] ] ], 
>  [ "30C", [ [  ] ] ] ];;

##
gap> create_classreps_slp:= function( classreps )
>     local words, l, len, lines, kerneloffset, inputoffset, cache, k,
>           found, pair, pos, diff, pos2, first, n, outputs, i, list;

>     # Find words for the products of kernel generators that occur.
>     words:= [];
>     for l in Set( Concatenation( List( classreps, x -> x[2] ) ) ) do
>       len:= Length( l );
>       if 2 <= len then
>         if not IsBound( words[ len ] ) then
>           words[ len ]:= [];
>         fi;
>         Add( words[ len ], l );
>       fi;
>     od;

>     lines:= [];
>     kerneloffset:= 60;
>     inputoffset:= 82;

>     # We have to form all products of length 2 of kernel generators.
>     cache:= [ [], [] ];
>     for l in words[2] do
>       Add( lines,
>            [ l[1] + kerneloffset, 1, l[2] + kerneloffset, 1 ] );
>       Add( cache[1], l );
>       Add( cache[2], Length( lines ) + inputoffset );
>     od;

>     # For products of length at least 3, we may use known products
>     # of length 2.  Longer matches are not considered.
>     for k in [ 3 .. Length( words ) ] do
>       for l in words[k] do
>         found:= false;
>         for pair in Combinations( l, 2 ) do
>           pos:= Position( cache[1], pair );
>           if pos <> fail then
>             diff:= Difference( l, pair );
>             if Length( diff ) = 1 then
>               Add( lines,
>                 [ cache[2][ pos ], 1, diff[1] + kerneloffset, 1 ] );
>             else
>               pos2:= Position( cache[1], diff );
>               if pos2 <> fail then
>                 Add( lines,
>                   [ cache[2][ pos ], 1, cache[2][ pos2 ], 1 ] );
>               else
>                 first:= cache[2][ pos ];
>                 for n in diff do
>                   Add( lines, [ first, 1, n + kerneloffset, 1 ] );
>                   first:= Length( lines ) + inputoffset;
>                 od;
>               fi;
>             fi;
>             Add( cache[1], l );
>             Add( cache[2], Length( lines ) + inputoffset );
>             found:= true;
>             break;
>           fi;
>         od;
>         if not found then
>           first:= l[1] + kerneloffset;
>           for n in l{ [ 2 .. Length( l ) ] } do
>             Add( lines, [ first, 1, n + kerneloffset, 1 ] );
>             first:= Length( lines ) + inputoffset;
>           od;
>           Add( cache[1], l );
>           Add( cache[2], Length( lines ) + inputoffset );
>         fi;
>       od;
>     od;

>     outputs:= [];

>     for i in [ 1 .. Length( classreps ) ] do
>       list:= classreps[i][2];
>       for l in list do
>         if l = [ 0 ] then
>           Add( outputs, [ kerneloffset + 1, 2 ] );
>         elif l = [] then
>           Add( outputs, [ i, 1 ] );
>         elif Length( l ) = 1 then
>           Add( lines, [ i, 1, l[1] + kerneloffset, 1 ] );
>           Add( outputs, [ Length( lines ) + inputoffset, 1 ] );
>         else
>           # The words are already cached.
>           pos:= Position( cache[1], l );
>           Add( lines, [ i, 1, cache[2][ pos ], 1 ] );
>           Add( outputs, [ Length( lines ) + inputoffset, 1 ] );
>         fi;
>       od;
>     od;

>     Add( lines, outputs );

>     return StraightLineProgram( lines, inputoffset );
> end;;
gap> slp_classreps:= create_classreps_slp( classrepsinfo );;

##
gap> kerperms:= ResultOfStraightLineProgram( slp_ker, stdperms );;
gap> co2classreps:= ResultOfStraightLineProgram(
>        slp_co2classreps.program, stdperms );;
gap> classreps:= ResultOfStraightLineProgram( slp_classreps,
>        Concatenation( co2classreps, kerperms ) );;

##
gap> g:= Group( stdperms );;
gap> SetConjugacyClasses( g,
>        List( classreps, x -> ConjugacyClass( g, x ) ) );
gap> libcb2b:= CharacterTable( "BM2" );
CharacterTable( "2^(1+22).Co2" )
gap> cen:= ClassPositionsOfCentre( libcb2b );;
gap> if CTblLib.IsMagmaAvailable() then
>      mgmt:= CharacterTableComputedByMagma( g, "2^22.Co2-Magma" );
>    else
>      mgmt:= libcb2b / cen;  # this is a hack ...
>    fi;

##
gap> cgens_2:= ResultOfStraightLineProgram( slp_maxes_2.program,
>                  GeneratorsOfGroup( b_2 ) );;
gap> cgens_5:= ResultOfStraightLineProgram( slp_maxes_2.program,
>                  GeneratorsOfGroup( b_5 ) );;
gap> cgens_2_std:= ResultOfStraightLineProgram( slp_co2, cgens_2 );;
gap> cgens_3_std:= ResultOfStraightLineProgram( slp_co2, cgens_3 );;
gap> cgens_5_std:= ResultOfStraightLineProgram( slp_co2, cgens_5 );;
gap> m:= GModuleByMats( cgens_5_std, GF(5) );;
gap> cf_5:= MTX.CompositionFactors( m );;
gap> SortParallel( List( cf_5, x -> x.dimension ), cf_5 );
gap> List( cf_5, x -> x.dimension );
[ 23, 2048, 2300 ]
gap> inputsB:= List( [ cgens_2_std, cgens_3_std, cgens_5_std ],
>      l -> Concatenation(
>             ResultOfStraightLineProgram( slp_co2classreps.program, l ),
>             ResultOfStraightLineProgram( slp_ker, l ) ) );;
gap> cf3std:= ResultOfStraightLineProgram( slp_co2, cf_3[2].generators );;
gap> inputs2048:= List( [ cf3std, cf_5[2].generators ],
>      l -> Concatenation(
>             ResultOfStraightLineProgram( slp_co2classreps.program, l ),
>             ResultOfStraightLineProgram( slp_ker, l ) ) );;

##
gap> slp:= RestrictOutputsOfSLP( slp_classreps, 1 );;
gap> centralinv_2048:= List( inputs2048,
>        l -> ResultOfStraightLineProgram( slp, l ) );;
gap> List( centralinv_2048, Order );
[ 2, 2 ]
gap> centralinv_B:= List( inputsB,
>        l -> ResultOfStraightLineProgram( slp, l ) );;

##
gap> cent_table:= rec( preimages:= [],
>                      fusionlabels:= [],
>                      projcharacter:= [] );;
gap> for i in [ 1 .. Length( classreps ) ] do
>      # identify the representative
>      slp:= RestrictOutputsOfSLP( slp_classreps, i );
>      id:= IdentifyClassName( inputsB[1], inputsB[2], inputsB[3], slp, fail );
>      order:= Int( Filtered( id, IsDigitChar ) );

>      if order mod 3 = 0 then
>        if order mod 5 = 0 then
>          # We cannot compute the Brauer character value.
>          value:= fail;
>        else
>          value:= BrauerCharacterValue(
>                      ResultOfStraightLineProgram( slp, inputs2048[2] ) );
>        fi;
>      else
>        value:= BrauerCharacterValue(
>                    ResultOfStraightLineProgram( slp, inputs2048[1] ) );
>      fi;

>      if value = 0 then
>        # Assume no splitting.
>        Add( cent_table.preimages, [ i ] );
>        Add( cent_table.fusionlabels, [ id ] );
>        Add( cent_table.projcharacter, value );
>      else
>        # Identify the class of the other preimage.
>        mats:= List( inputsB,
>                     l -> ResultOfStraightLineProgram( slp, l ) );
>        id2:= IdentifyClassName( mats[1] * centralinv_B[1],
>                            mats[2] * centralinv_B[2],
>                            mats[3] * centralinv_B[3],
>                            fail, fail );
>        if value = fail then
>          # no Brauer character value known
>          Add( cent_table.preimages, [ i ] );
>          Add( cent_table.fusionlabels, [ id, id2 ] );
>          Add( cent_table.projcharacter, value );
>        else
>          # two preimage classes, take the positive value first
>          Add( cent_table.preimages, [ i, i ] );
>          if value > 0 then
>            Add( cent_table.fusionlabels, [ id, id2 ] );
>            Add( cent_table.projcharacter, value );
>          else
>            Add( cent_table.fusionlabels, [ id2, id ] );
>            Add( cent_table.projcharacter, -value );
>          fi;
>        fi;
>      fi;
>    od;

##
gap> chi:= cent_table.projcharacter;;
gap> failpos:= Positions( chi, fail );;
gap> OrdersClassRepresentatives( mgmt ){ failpos };
[ 15, 30, 30, 30, 15, 15, 30, 30, 30, 30, 30, 30 ]
gap> SizesCentralizers( mgmt ){ failpos };
[ 120, 120, 120, 120, 30, 30, 120, 120, 120, 120, 30, 30 ]
gap> used:= 0;;
gap> for i in[ 1 .. 388 ] do
>      if chi[i] <> 0 and chi[i] <> fail then
>        used:= used + 2 * SizesConjugacyClasses( mgmt )[i] * chi[i]^2;
>      fi;
>    od;
gap> used / ( 2 * Size( mgmt ) ) > 1 - 7^2 / 120;
true

##
gap> for i in failpos do
>      v:= ChineseRem( [ 3, 5 ], [ chi[ PowerMap( mgmt, 3, i ) ],
>                                  chi[ PowerMap( mgmt, 5, i ) ] ] );
>     if v > 7 then
>       v:= v - 15;
>     fi;
>     chi[i]:= AbsInt( v );
>   od;
gap> chi{ failpos };
[ 2, 0, 0, 0, 1, 1, 2, 0, 0, 0, 1, 1 ]

##
gap> split:= Filtered( failpos, x -> chi[x] <> 0 );
[ 343, 347, 348, 383, 387, 388 ]
gap> nonsplit:= Difference( failpos, split );
[ 344, 345, 346, 384, 385, 386 ]
gap> pow:= PowerMap( mgmt, 3 ){ split };
[ 138, 136, 136, 263, 261, 261 ]
gap> chi{ split };
[ 2, 1, 1, 2, 1, 1 ]
gap> chi{ pow };
[ 8, 2, 2, 4, 2, 2 ]
gap> cent_table.fusionlabels{ split };
[ [ "15A", "30D" ], [ "15B", "30GH" ], [ "15B", "30GH" ], [ "30A", "30E" ], 
  [ "30GH", "30F" ], [ "30GH", "30F" ] ]
gap> cent_table.fusionlabels{ pow };
[ [ "5A", "10B" ], [ "10D", "5B" ], [ "10D", "5B" ], [ "10A", "10E" ], 
  [ "10F", "10D" ], [ "10F", "10D" ] ]

##
gap> First( powerinfo, l -> l[1] = "30A" );
[ "30A", [ [ 2, "15A" ], [ 3, "10A" ], [ 5, "6B" ] ] ]
gap> First( powerinfo, l -> l[1] = "30F" );
[ "30F", [ [ 2, "15B" ], [ 3, "10F" ], [ 5, "6I" ] ] ]
gap> cent_table.fusionlabels[ split[4] ]:= Permuted(
>        cent_table.fusionlabels[ split[4] ], (1,2) );;
gap> for i in nonsplit do
>      Unbind( cent_table.fusionlabels[i][2] );
>    od;

##
gap> for i in split do
>      cent_table.preimages[i]:= [ i, i ];
>    od;
gap> factorfusion:= Concatenation( cent_table.preimages );;
gap> cb2b:= rec( UnderlyingCharacteristic:= 0,
>                Size:= 2 * Size( mgmt ),
>                Identifier:= "C_B(2B)",
>                SizesCentralizers:= [] );;
gap> proj:= InverseMap( factorfusion );;
gap> for i in [ 1 .. Length( proj ) ] do
>      if IsInt( proj[i] ) then
>        cb2b.SizesCentralizers[ proj[i] ]:= SizesCentralizers( mgmt )[i];
>      else
>        cb2b.SizesCentralizers{ proj[i] }:=
>            SizesCentralizers( mgmt )[i] * [ 2, 2 ];
>      fi;
>    od;

##
gap> fusionlabels:= Concatenation( cent_table.fusionlabels );;
gap> cb2b.OrdersClassRepresentatives:= List(
>        Concatenation( cent_table.fusionlabels ),
>        x -> Int( Filtered( x, IsDigitChar ) ) );;
gap> ConvertToCharacterTable( cb2b );;

##
gap> chi_c:= [];;
gap> for i in [ 1 .. Length( chi ) ] do
>      if chi[i] = 0 then
>        chi_c[ proj[i] ]:= 0;
>      else
>        chi_c{ proj[i] }:= [ 1, -1 ] * chi[i];
>      fi;
>    od;
gap> factirr:= List( Irr( mgmt ), x -> x{ factorfusion } );;

##
gap> factirr_co2:= Filtered( factirr,
>                      x -> ClassPositionsOfKernel( x ) <> [ 1, 2 ] );;
gap> Length( factirr_co2 );
60
gap> ten:= Tensored( factirr_co2, [ chi_c ] );;
gap> Set( List( ten, x -> ScalarProduct( cb2b, x, x ) ) );
[ 1 ]
gap> irr:= Concatenation( factirr, ten );;
gap> Size( cb2b ) = Sum( List( irr, x -> x[1]^2 ) );
true
gap> SetIrr( cb2b, List( irr, x -> Character( cb2b, x ) ) );

##
gap> powermaps:= ComputedPowerMaps( cb2b );
[  ]
gap> for p in Set( Factors( Size( cb2b ) ) ) do
>      init:= CompositionMaps( InverseMap( factorfusion ),
>          CompositionMaps( PowerMap( mgmt, p ), factorfusion ) );
>      poss:= PossiblePowerMaps( cb2b, p, rec( powermap:= init ) );
>      if Length( poss ) <> 1 then
>        Error( Ordinal( p ), " power map is not unique" );
>      fi;
>      powermaps[p]:= poss[1];
>    od;

##
gap> libtbl:= CharacterTable( "BM2" );;
gap> Set( Irr( libtbl ) ) = Set( Irr( cb2b ) );
true
gap> IsRecord( TransformingPermutationsCharacterTables( libtbl, cb2b ) );
true

##
gap> libB:= CharacterTable( "B" );;
gap> libBnames:= ClassNames( libB, "ATLAS" );;
gap> Bclassinfo:= rec();;
gap> SetCentralizerOrder:= function( classname, value )
>        local pos;

>        pos:= Position( libBnames, classname );
>        if pos = fail then
>          Print( "no class name '", classname, "'?" );
>          return false;
>        elif SizesCentralizers( libB )[ pos ] <> value then
>          Error( "wrong centralizer order!" );
>        fi;
>        Bclassinfo.( classname ):=
>            [ Int( Filtered( classname, IsDigitChar ) ), value ];
>        return true;
>    end;;

##
gap> SetCentralizerOrder( "1A", Size( libB ) );;
gap> SetCentralizerOrder( "11A", 11 * 120 );;
gap> SetCentralizerOrder( "13A", 13 * 24 );;
gap> SetCentralizerOrder( "17A", 17 * 4 );;
gap> SetCentralizerOrder( "19A", 19 * 2 );;
gap> SetCentralizerOrder( "23A", 23 * 2 );;
gap> SetCentralizerOrder( "23B", 23 * 2 );;
gap> SetCentralizerOrder( "31A", 31 );;
gap> SetCentralizerOrder( "31B", 31 );;
gap> SetCentralizerOrder( "47A", 47 );;
gap> SetCentralizerOrder( "47B", 47 );;

##
gap> RootInfoFromLabels:= function( label )
>     local found, exp, res, entry, pos, root, ord;

>     found:= [ label ];
>     exp:= [ Int( Filtered( label, IsDigitChar ) ) ];
>     res:= rec( total:= [], labels:= [] );
>     res.total[ exp[1] ]:= 1;
>     res.labels[ exp[1] ]:= [ label ];
>     for entry in powerinfo do
>       for root in entry[2] do
>         if not entry[1] in found then
>           pos:= Position( found, root[2] );
>           if pos <> fail then
>             ord:= exp[ pos ] * root[1];
>             Add( exp, ord );
>             Add( found, entry[1] );
>             if not IsBound( res.total[ ord ] ) then
>               res.total[ ord ]:= 0;
>               res.labels[ ord ]:= [];
>             fi;
>             res.total[ ord ]:= res.total[ ord ] + 1;
>             Add( res.labels[ ord ], entry[1] );
>           fi;
>         fi;
>       od;
>     od;
>     return res;
> end;;

##
gap> RootInfoFromTable:= function( tbl, pos )
>     local orders, posord, res, i, ord;

>     orders:= OrdersClassRepresentatives( tbl );
>     posord:= orders[ pos ];
>     res:= rec( total:= [], classpos:= [] );
>     for i in [ 1 .. NrConjugacyClasses( tbl ) ] do
>       ord:= orders[i];
>       if ord mod posord = 0 and
>          PowerMap( tbl, ord / posord, i ) = pos then
>         if not IsBound( res.total[ ord ] ) then
>           res.total[ ord ]:= 0;
>           res.classpos[ ord ]:= [];
>         fi;
>         res.total[ ord ]:= res.total[ ord ] + 1;
>         Add( res.classpos[ ord ], i );
>       fi;
>     od;
>     return res;
> end;;

##
gap> norm2A:= CharacterTable( "2.2E6(2).2" );;
gap> pos2A:= ClassPositionsOfCentre( norm2A );
[ 1, 2 ]
gap> rootinfo2A_t:= RootInfoFromTable( norm2A, pos2A[2] );;
gap> pos2B:= ClassPositionsOfCentre( cb2b );
[ 1, 2 ]
gap> rootinfo2B_t:= RootInfoFromTable( cb2b, pos2B[2] );;
gap> norm2C:= CharacterTableOfIndexTwoSubdirectProduct(
>      CharacterTable( "F4(2)" ), CharacterTable( "F4(2).2" ),
>      CharacterTable( "2^2" ), CharacterTable( "D8" ), "norm2C" );;
gap> pos2C:= ClassPositionsOfCentre( norm2C.table );
[ 1, 2 ]
gap> rootinfo2C_t:= RootInfoFromTable( norm2C.table, pos2C[2] );;
gap> sup_norm2D:= CharacterTable( "BM4" );
CharacterTable( "2^(9+16).S8(2)" )
gap> pos2D:= Positions( SizesCentralizers( sup_norm2D ), 11689182992793600 );
[ 4 ]
gap> rootinfo2D_t:= RootInfoFromTable( sup_norm2D, pos2D[1] );;
gap> rootinfo2A_l:= RootInfoFromLabels( "2A" );;
gap> rootinfo2B_l:= RootInfoFromLabels( "2B" );;
gap> rootinfo2C_l:= RootInfoFromLabels( "2C" );;
gap> rootinfo2D_l:= RootInfoFromLabels( "2D" );;

##
gap> List( [ rootinfo2A_t, rootinfo2B_t, rootinfo2C_t, rootinfo2D_t ],
>          r -> Filtered( [ 52, 56, 70 ], i -> IsBound( r.total[i] ) ) );
[ [ 70 ], [ 56 ], [ 52 ], [  ] ]
gap> List( [ rootinfo2A_l, rootinfo2B_l, rootinfo2C_l, rootinfo2D_l ],
>          r -> Filtered( [ 52, 56, 70 ], i -> IsBound( r.total[i] ) ) );
[ [ 70 ], [ 56 ], [ 52 ], [  ] ]

##
gap> List( [ rootinfo2A_t, rootinfo2B_t, rootinfo2C_t, rootinfo2D_t ],
>          r -> Sum( Compacted( r.total ) ) );
[ 20, 83, 14, 40 ]
gap> List( [ rootinfo2A_l, rootinfo2B_l, rootinfo2C_l, rootinfo2D_l ],
>          r -> Sum( Compacted( r.total ) ) );
[ 19, 77, 14, 40 ]

##
gap> rootinfo2A_t.total[34];
2
gap> rootinfo2A_l.total[34];
1
gap> rootinfo2A_l.labels[34];
[ "34BC" ]
gap> pos34:= rootinfo2A_t.classpos[34];
[ 133, 135 ]
gap> PowerMap( norm2A, 3 ){ pos34 };
[ 135, 133 ]

##
gap> rootinfo2B_t.total{ [ 16, 30, 32, 46, 56 ] };
[ 6, 3, 4, 2, 2 ]
gap> rootinfo2B_l.total{ [ 16, 30, 32, 46, 56 ] };
[ 5, 2, 2, 1, 1 ]

##
gap> rootinfo2B_l.labels[30];
[ "30D", "30GH" ]
gap> pos30:= rootinfo2B_t.classpos[30];
[ 388, 393, 395 ]
gap> PowerMap( cb2b, 7 ){ pos30 };
[ 388, 395, 393 ]
gap> List( pos30,
>          i -> Number( PowerMap( cb2b, 2 ), x -> x = i ) );
[ 2, 0, 0 ]
gap> List( rootinfo2B_l.labels[30],
>          l -> Length( RootInfoFromLabels( l ).total ) );
[ 60, 30 ]

##
gap> rootinfo2B_l.labels[32];
[ "32AB", "32CD" ]
gap> pos32:= rootinfo2B_t.classpos[32];
[ 399, 400, 403, 404 ]
gap> PowerMap( cb2b, 5 ){ pos32 };
[ 400, 399, 404, 403 ]

##
gap> rootinfo2B_l.labels[46];
[ "46AB" ]

##
gap> rootinfo2B_l.labels[56];
[ "56AB" ]
gap> pos56:= rootinfo2B_t.classpos[56];
[ 437, 438 ]
gap> PowerMap( cb2b, 3 ){ pos56 };
[ 437, 438 ]
gap> PowerMap( cb2b, 5 ){ pos56 };
[ 438, 437 ]

##
gap> pos:= rootinfo2B_t.classpos[16];
[ 233, 234, 235, 251, 252, 258 ]
gap> PowerMap( cb2b, 2 ){ pos };
[ 69, 69, 69, 104, 104, 104 ]
gap> List( rootinfo2B_t.classpos[16],
>          i -> Number( PowerMap( cb2b, 2 ), x -> x = i ) );
[ 0, 0, 0, 2, 0, 2 ]
gap> rootinfo2B_l.labels[16];
[ "16A", "16B", "16E", "16DF", "16C" ]
gap> Filtered( powerinfo, l -> l[1] in rootinfo2B_l.labels[16] );
[ [ "16A", [ [ 2, "8D" ] ] ], [ "16B", [ [ 2, "8D" ] ] ], 
  [ "16E", [ [ 2, "8D" ] ] ], [ "16DF", [ [ 2, "8H" ] ] ], 
  [ "16C", [ [ 2, "8H" ] ] ] ]
gap> List( rootinfo2B_l.labels[16],
>          l -> IsBound( RootInfoFromLabels( l ).total[32] ) );
[ false, false, false, true, true ]

##
gap> tosplit:= List( Filtered( powerinfo,
>                              x -> Number( x[1], IsAlphaChar ) > 1 ),
>                    x -> x[1] );
[ "16DF", "23AB", "30GH", "31AB", "32AB", "32CD", "34BC", "46AB", "47AB", 
  "56AB" ]
gap> for l in tosplit do
>      ord:= Filtered( l, IsDigitChar );
>      new:= List( Filtered( l, IsAlphaChar ),
>                  c -> Concatenation( ord, [ c ] ) );
>      pos:= PositionProperty( powerinfo, x -> x[1] = l );
>      Append( powerinfo, List( new, x -> [ x,
>                  StructuralCopy( powerinfo[ pos ][2] ) ] ) );
>      Unbind( powerinfo[ pos ] );
>    od;
gap> powerinfo:= Compacted( powerinfo );;
gap> SortParallel(
>      List( powerinfo,
>            x -> [ Int( Filtered( x[1], IsDigitChar ) ), x[1] ] ),
>      powerinfo );

##
gap> Filtered( powerinfo, x -> ForAny( x[2], p -> p[2] in tosplit ) );
[ [ "32C", [ [ 2, "16DF" ] ] ], [ "32D", [ [ 2, "16DF" ] ] ], 
  [ "46A", [ [ 2, "23AB" ], [ 23, "2B" ] ] ], 
  [ "46B", [ [ 2, "23AB" ], [ 23, "2B" ] ] ] ]
gap> entry:= First( powerinfo, x -> x[1] = "32C" );;
gap> entry[2][1][2]:= "16D";;
gap> entry:= First( powerinfo, x -> x[1] = "32D" );;
gap> entry[2][1][2]:= "16D";;
gap> entry:= First( powerinfo, x -> x[1] = "46A" );;
gap> entry[2][1][2]:= "23A";;
gap> entry:= First( powerinfo, x -> x[1] = "46B" );;
gap> entry[2][1][2]:= "23B";;

##
gap> Length( powerinfo );
184
gap> Bnames:= List( powerinfo, x -> x[1] );;

##
gap> n3a:= CharacterTable( "S3xFi22.2" );;
gap> pos:= ClassPositionsOfPCore( n3a, 3 );
[ 1, 113 ]
gap> rootinfo3A_t:= RootInfoFromTable( n3a, pos[2] );;
gap> rootinfo3A_l:= RootInfoFromLabels( "3A" );;
gap> n3b:= CharacterTable( "3^(1+8).2^(1+6).U4(2).2" );;
gap> pos:= ClassPositionsOfPCore( n3b, 3 );
[ 1 .. 4 ]
gap> Filtered( ClassPositionsOfNormalSubgroups( n3b ),
>              n -> IsSubset( pos, n ) );
[ [ 1 ], [ 1, 2 ], [ 1 .. 4 ] ]
gap> rootinfo3B_t:= RootInfoFromTable( n3b, pos[2] );;
gap> rootinfo3B_l:= RootInfoFromLabels( "3B" );;
gap> IsBound( rootinfo3A_t.total[66] );
true
gap> IsBound( rootinfo3B_t.total[66] );
false
gap> rootinfo3A_t.total = rootinfo3A_l.total;
true
gap> rootinfo3B_t.total = rootinfo3B_l.total;
true
gap> n5a:= CharacterTable( "5:4xHS.2" );
CharacterTable( "5:4xHS.2" )
gap> pos:= ClassPositionsOfPCore( n5a, 5 );
[ 1, 40 ]
gap> rootinfo5A_t:= RootInfoFromTable( n5a, pos[2] );;
gap> rootinfo5A_l:= RootInfoFromLabels( "5A" );;
gap> n5b:= CharacterTable( "5^(1+4).2^(1+4).A5.4" );
CharacterTable( "5^(1+4).2^(1+4).A5.4" )
gap> pos:= ClassPositionsOfPCore( n5b, 5 );
[ 1 .. 4 ]
gap> Filtered( ClassPositionsOfNormalSubgroups( n5b ),
>              n -> IsSubset( pos, n ) );
[ [ 1 ], [ 1, 2 ], [ 1 .. 4 ] ]
gap> rootinfo5B_t:= RootInfoFromTable( n5b, pos[2] );;
gap> rootinfo5B_l:= RootInfoFromLabels( "5B" );;
gap> IsBound( rootinfo5A_t.total[70] );
true
gap> IsBound( rootinfo5B_t.total[70] );
false
gap> rootinfo5A_t.total = rootinfo5A_l.total;
true
gap> rootinfo5B_t.total = rootinfo5B_l.total;
true

##
gap> powerinforec:= rec();;
gap> for entry in powerinfo do
>      powerinforec.( entry[1] ):= entry[2];
>    od;

##
gap> rootinfo2A_l:= RootInfoFromLabels( "2A" );;
gap> rootinfo2B_l:= RootInfoFromLabels( "2B" );;
gap> rootinfo2C_l:= RootInfoFromLabels( "2C" );;
gap> rootinfo2D_l:= RootInfoFromLabels( "2D" );;
gap> rootinfo2A_t.total = rootinfo2A_l.total;
true
gap> rootinfo2B_t.total = rootinfo2B_l.total;
true
gap> rootinfo2C_t.total = rootinfo2C_l.total;
true
gap> rootinfo2D_t.total = rootinfo2D_l.total;
true

##
gap> IdentifyCentralizerOrders:= function( normtbl, rl, rt )
>     local n, identified, found, i, unknown, class, d, linfo, p, e, cand,
>           imgs, im, pos, powerlabel, dd, cent;
>     n:= First( [ 1 .. Length( rl ) ], i -> IsBound( rl[i] ) );
>     identified:= [ [], [] ];
>     found:= true;
>     while found do
>       found:= false;
>       for i in [ 1 .. Length( rl ) ] do
>       if IsBound( rl[i] ) then
>         unknown:= Difference( rl[i], identified[1] );
>         if Length( unknown ) = 1 then
>           # Identify the class.
>           class:= Difference( rt[i], identified[2] )[1];
>           Add( identified[1], unknown[1] );
>           Add( identified[2], class );
>           found:= true;
>           # Identify the admissible powers.
>           for d in Difference( DivisorsInt( i / n ), [ 1 ] ) do
>             linfo:= powerinforec.( unknown[1] );
>             for p in Factors( d ) do
>               e:= First( linfo, x -> x[1] = p );
>               linfo:= powerinforec.( e[2] );
>             od;
>             if not e[2] in identified[1] then
>               Add( identified[1], e[2] );
>               Add( identified[2], PowerMap( normtbl, d, class ) );
>               found:= true;
>             fi;
>           od;
>         else
>           # Try to identify roots whose powers are identified.
>           for d in Difference( DivisorsInt( i / n ), [ 1 ] ) do
>             cand:= Difference( rt[i], identified[2] );
>             imgs:= PowerMap( normtbl, d ){ cand };
>             for im in Intersection( imgs, identified[2] ) do
>               pos:= Positions( imgs, im );
>               if Length( pos ) = 1 then
>                 class:= cand[ pos[1] ];
>                 powerlabel:= identified[1][
>                                  Position( identified[2], im ) ];
>                 # Find the labels of the 'd'-th powers of 'unknown'.
>                 linfo:= List( unknown, l -> powerinforec.( l ) );
>                 for p in Factors( d ) do
>                   e:= List( linfo, ll -> First( ll, x -> x[1] = p ) );
>                   linfo:= List( e, ee -> powerinforec.( ee[2] ) );
>                 od;
>                 linfo:= List( e, x -> x[2] );
>                 pos:= Position( linfo, powerlabel );
>                 Add( identified[1], unknown[ pos ] );
>                 Add( identified[2], class );
>                 # Identify the admissible powers.
>                 for dd in Difference( DivisorsInt( i / n ), [ 1 ] ) do
>                   linfo:= powerinforec.( unknown[ pos ] );
>                   for p in Factors( dd ) do
>                     e:= First( linfo, x -> x[1] = p );
>                     linfo:= powerinforec.( e[2] );
>                   od;
>                   if not e[2] in identified[1] then
>                     Add( identified[1], e[2] );
>                     Add( identified[2], PowerMap( normtbl, dd, class ) );
>                     found:= true;
>                   fi;
>                 od;
>                 found:= true;
>                 break; # since we have to update 'unknown'
>               fi;
>             od;
>             if found then
>               break; # since we have to update 'unknown'
>             fi;
>           od;
>         fi;
>       fi;
>     od;
>   od;
>   # Where the centralizer order is unique, set it.
>   for i in [ 1 .. Length( rl ) ] do
>     if IsBound( rl[i] ) then
>       cand:= Difference( rt[i], identified[2] );
>       cent:= Set( SizesCentralizers( normtbl ){ cand } );
>       if Length( cent ) = 1 then
>         Append( identified[1], Difference( rl[i], identified[1] ) );
>         Append( identified[2], cand );
>       fi;
>     fi;
>   od;
>   # Set the centralizer orders.
>   for i in [ 1 .. Length( identified[1] ) ] do
>     if not IsBound( Bclassinfo.( identified[1][i] ) ) then
>       Print( "#I  identify ", identified[1][i], "\n" );
>     fi;
>     SetCentralizerOrder( identified[1][i],
>         SizesCentralizers( normtbl )[ identified[2][i] ] );
>   od;
>   # Return the information about unidentified classes.
>   return [ Difference( Concatenation( Compacted( rl ) ), identified[1] ),
>            Difference( Concatenation( Compacted( rt ) ), identified[2] ) ];
> end;;

##
gap> IdentifyCentralizerOrders( norm2A,
>        rootinfo2A_l.labels, rootinfo2A_t.classpos );
#I  identify 2A
#I  identify 10A
#I  identify 22A
#I  identify 26B
#I  identify 38A
#I  identify 66A
#I  identify 6A
#I  identify 70A
#I  identify 14A
#I  identify 14B
#I  identify 42A
#I  identify 6B
#I  identify 6D
#I  identify 42B
#I  identify 30B
#I  identify 30A
#I  identify 34B
#I  identify 34C
[ [ "18A", "18B" ], [ 74, 76 ] ]
gap> for i in Union( rootinfo2B_t.classpos ) do
>      if IsBound( powerinforec.( fusionlabels[i] ) ) then
>        SetCentralizerOrder( fusionlabels[i],
>            SizesCentralizers( cb2b )[i] );
>      fi;
>    od;
gap> IdentifyCentralizerOrders( cb2b,
>        rootinfo2B_l.labels, rootinfo2B_t.classpos );
#I  identify 30G
#I  identify 30H
#I  identify 32A
#I  identify 32B
#I  identify 32C
#I  identify 32D
#I  identify 46A
#I  identify 46B
#I  identify 56A
#I  identify 56B
--> --------------------

--> maximum size reached

--> --------------------

[ Dauer der Verarbeitung: 0.13 Sekunden  (vorverarbeitet)  ]