knotkit/main.cpp

2143 lines
51 KiB
C++
Raw Normal View History

#include <knotkit.h>
// test for ring
template<class R> void
test_ring (int p)
{
R zero (0);
R one (1);
R minus_one (-1);
assert (zero == 0);
assert (zero | zero);
assert (one | zero);
assert (minus_one | zero);
assert (! (zero | one));
assert (! (zero | minus_one));
assert (one.is_unit ());
assert (minus_one.is_unit ());
assert (one.recip () == one);
assert (minus_one.recip () == minus_one);
if (p)
assert (R (p) == 0);
if (p != 2)
assert (one != minus_one);
int n = (p
? std::min (p, 20)
: 20);
for (int i = -n; i <= n; i ++)
{
R x (i);
if (x.is_unit ())
{
assert (x * x.recip () == one);
assert (x.recip () * x == one);
assert (x.recip ().recip () == x);
}
assert (one | x);
assert (minus_one | x);
if (x != 0)
{
assert (x | zero);
assert (! (zero | x));
}
for (int j = -n; j <= n; j ++)
{
R y (j);
assert (- (-x) == x);
assert (x + y == y + x);
assert (x * y == y * x);
if (x != 0 && x | y)
{
R q = y.divide_exact (x);
assert (y == q * x);
}
if (x != 0 || y != 0)
{
tuple<R, R, R> t = x.extended_gcd (y);
assert (get<0> (t) == get<1> (t)*x + get<2> (t)*y);
}
for (int k = -n; k <= n; k ++)
{
R z (k);
assert ((x + y) + z == x + (y + z));
assert ((x * y) * z == x * (y * z));
assert (x*(y + z) == x*y + x*z);
assert ((x + y)*z == x*z + y*z);
}
}
}
}
template<class F> void
test_field ()
{
for (unsigned i = 1; i <= 8; i ++)
for (unsigned j = 1; j <= rolfsen_crossing_knots (i); j ++)
{
knot_diagram kd (rolfsen_knot (i, j));
show (kd); newline ();
cube<F> c (kd);
mod_map<F> d = c.compute_d (1, 0, 0, 0, 0);
assert (d.compose (d) == 0);
ptr<const quotient_module<F> > H = d.homology ();
display ("H:\n", *H);
chain_complex_simplifier<F> s (c.khC, d, 1);
display ("s.new_C:\n", *s.new_C);
assert (H->dim () == s.new_C->dim ());
}
}
bool
rank_lte (multivariate_laurentpoly<Z> p,
multivariate_laurentpoly<Z> q)
{
for (map<multivariate_laurent_monomial, Z>::const_iter i = p.coeffs; i; i ++)
{
Z a = i.val ();
Z b = q.coeffs(i.key (), Z (0));
assert (a != 0 && b != 0);
if (a > b)
return 0;
}
return 1;
}
triple<multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z> >
square (knot_diagram &kd)
{
cube<Z2> c (kd);
mod_map<Z2> d = c.compute_d (1, 0, 0, 0, 0);
chain_complex_simplifier<Z2> s (c.khC, d, 1);
steenrod_square sq (c, d, s);
mod_map<Z2> sq1 = sq.sq1 ();
// display ("sq1:\n", sq1);
mod_map<Z2> sq2 = sq.sq2 ();
// display ("sq2:\n", sq2);
assert (sq1.compose (sq1) == 0);
assert (sq2.compose (sq2) + sq1.compose (sq2).compose (sq1) == 0);
multivariate_laurentpoly<Z> P = s.new_C->free_poincare_polynomial ();
ptr<const free_submodule<Z2> > sq1_im = sq1.image ();
multivariate_laurentpoly<Z> sq1_P = sq1_im->free_poincare_polynomial ();
ptr<const free_submodule<Z2> > sq2_im = sq2.image ();
multivariate_laurentpoly<Z> sq2_P = sq2_im->free_poincare_polynomial ();
return triple<multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z> > (P, sq1_P, sq2_P);
}
void
compute_show_kh_sq (knot_desc desc
#if 0
,
multivariate_laurentpoly<Z> orig_P,
multivariate_laurentpoly<Z> orig_sq1_P,
multivariate_laurentpoly<Z> orig_sq2_P
#endif
)
{
knot_diagram kd = desc.diagram ();
printf ("computing %s...\n", kd.name.c_str ());
cube<Z2> c (kd);
mod_map<Z2> d = c.compute_d (1, 0, 0, 0, 0);
chain_complex_simplifier<Z2> s (c.khC, d, 1);
steenrod_square sq (c, d, s);
mod_map<Z2> sq1 = sq.sq1 ();
mod_map<Z2> sq2 = sq.sq2 ();
assert (sq1.compose (sq1) == 0);
assert (sq2.compose (sq2) + sq1.compose (sq2).compose (sq1) == 0);
multivariate_laurentpoly<Z> P = s.new_C->free_poincare_polynomial ();
ptr<const free_submodule<Z2> > sq1_im = sq1.image ();
multivariate_laurentpoly<Z> sq1_P = sq1_im->free_poincare_polynomial ();
ptr<const free_submodule<Z2> > sq2_im = sq2.image ();
multivariate_laurentpoly<Z> sq2_P = sq2_im->free_poincare_polynomial ();
printf (" P "); display (P);
printf (" sq1_P "); display (sq1_P);
printf (" sq2_P "); display (sq2_P);
#if 0
assert (P == orig_P);
assert (sq1_P == orig_sq1_P);
assert (sq2_P == orig_sq2_P);
#endif
}
unsigned
homological_width (ptr<const module<Z2> > H)
{
int maxd = -1000,
mind = 1000;
set<grading> gs = H->gradings ();
for (set_const_iter<grading> gg = gs; gg; gg ++)
{
grading hq = gg.val ();
int d = 2 * hq.h - hq.q;
if (d < mind)
mind = d;
if (d > maxd)
maxd = d;
}
int dwidth = maxd - mind;
assert (is_even (dwidth));
unsigned hwidth = (dwidth / 2) + 1;
return hwidth;
}
basedvector<unsigned, 1> hwidth_knots;
void
load (map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > &knot_kh_sq,
knot_desc desc)
{
char buf[1000];
if (desc.t == knot_desc::TORUS)
{
sprintf (buf, "knot_kh_sq/T.dat");
}
else
{
unsigned j0 = desc.j;
switch (desc.t)
{
case knot_desc::ROLFSEN:
sprintf (buf, "knot_kh_sq/%d_%d.dat", desc.i, j0);
break;
case knot_desc::HTW:
sprintf (buf, "knot_kh_sq/K%d_%d.dat", desc.i, j0);
break;
case knot_desc::MT:
sprintf (buf, "knot_kh_sq/L%d_%d.dat", desc.i, j0);
break;
default: abort ();
}
}
struct stat stat_buf;
if (stat (buf, &stat_buf) != 0)
{
if (errno == ENOENT)
return;
stderror ("stat: %s", buf);
exit (EXIT_FAILURE);
}
printf ("loading %s...\n", buf);
file_reader r (buf, 1);
map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > m (r);
for (map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > >::const_iter i = m; i; i ++)
{
mod_map<Z2> sq1 = i.val ().first;
ptr<const module<Z2> > H = sq1.domain ();
unsigned hwidth = homological_width (H);
// hwidth_knots[hwidth] ++;
#if 0
if (hwidth == 2)
continue;
if (i.key ().t == knot_desc::MT
&& i.key ().diagram ().num_components () == 1)
continue;
#endif
knot_kh_sq.push (i.key (), i.val ());
}
printf ("done.\n");
}
void
load (map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > &knot_kh_sq,
const char *file)
{
char buf[1000];
sprintf (buf, "knot_kh_sq/%s", file);
printf ("loading %s...\n", buf);
struct stat stat_buf;
if (stat (buf, &stat_buf) != 0)
{
if (errno == ENOENT)
{
printf ("%s does not exist.\n", buf);
return;
}
stderror ("stat: %s", buf);
exit (EXIT_FAILURE);
}
file_reader r (buf, 1);
map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > m (r);
for (map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > >::const_iter i = m; i; i ++)
{
#if 0
mod_map<Z2> sq1 = i.val ().first;
ptr<const module<Z2> > H = sq1.domain ();
unsigned hwidth = homological_width (H);
// hwidth_knots[hwidth] ++;
#endif
#if 0
if (hwidth == 2)
continue;
if (i.key ().t == knot_desc::MT
&& i.key ().diagram ().num_components () == 1)
continue;
#endif
knot_kh_sq.push (i.key (), i.val ());
}
printf ("done.\n");
}
static const int block_size = 100;
void
sage_show (FILE *fp,
std::string prefix,
map<grading, basedvector<unsigned, 1> > hq_gens,
ptr<const module<Z2> > H)
{
fprintf (fp, "%s_hom={", prefix.c_str ());
bool first = 1;
for (map<grading, basedvector<unsigned, 1> >::const_iter i = hq_gens; i; i ++)
{
if (first)
first = 0;
else
fprintf (fp, ", ");
fprintf (fp, "(%d, %d): %d", i.key ().h, i.key ().q, i.val ().size ());
}
fprintf (fp, "}\n");
}
void
sage_show (FILE *fp,
std::string prefix,
map<grading, basedvector<unsigned, 1> > hq_gens,
std::string name,
grading delta,
mod_map<Z2> f)
{
fprintf (fp, "%s_%s={", prefix.c_str (), name.c_str ());
bool first = 1;
for (map<grading, basedvector<unsigned, 1> >::const_iter i = hq_gens; i; i ++)
{
grading from_hq = i.key ();
basedvector<unsigned, 1> from_gens = i.val ();
grading to_hq = from_hq + delta;
if (hq_gens % to_hq)
{
basedvector<unsigned, 1> to_gens = hq_gens(to_hq);
if (first)
first = 0;
else
fprintf (fp, ", ");
fprintf (fp, "(%d, %d): [", from_hq.h, from_hq.q);
bool first2 = 1;
for (unsigned j = 1; j <= from_gens.size (); j ++)
{
unsigned gj = from_gens[j];
if (first2)
first2 = 0;
else
fprintf (fp, ", ");
fprintf (fp, "(");
for (unsigned k = 1; k <= to_gens.size (); k ++)
{
unsigned gk = to_gens[k];
if (k > 1)
fprintf (fp, ", ");
if (f[gj](gk) == 1)
fprintf (fp, "1");
else
{
assert (f[gj](gk) == 0);
fprintf (fp, "0");
}
}
fprintf (fp, ")");
}
fprintf (fp, "]");
}
}
fprintf (fp, "}\n");
}
void
sage_show_khsq (FILE *fp,
std::string prefix,
ptr<const module<Z2> > H,
mod_map<Z2> sq1,
mod_map<Z2> sq2)
{
unsigned n = H->dim ();
map<grading, basedvector<unsigned, 1> > hq_gens;
for (unsigned i = 1; i <= H->dim (); i ++)
hq_gens[H->generator_grading (i)].append (i);
unsigned t = 0;
for (map<grading, basedvector<unsigned, 1> >::const_iter i = hq_gens; i; i ++)
t += i.val ().size ();
assert (t == n);
sage_show (fp, prefix, hq_gens, H);
sage_show (fp, prefix, hq_gens, "sq1", grading (1, 0), sq1);
sage_show (fp, prefix, hq_gens, "sq2", grading (2, 0), sq2);
fprintf (fp, "\n");
}
void
test_sage_show ()
{
knot_diagram kd (mt_link (11, 0, 449));
show (kd); newline ();
planar_diagram pd (kd);
pd.display_knottheory ();
cube<Z2> c (kd);
mod_map<Z2> d = c.compute_d (1, 0, 0, 0, 0);
chain_complex_simplifier<Z2> s (c.khC, d, 1);
steenrod_square sq (c, d, s);
mod_map<Z2> sq1 = sq.sq1 ();
mod_map<Z2> sq2 = sq.sq2 ();
// display ("sq2:\n", sq2);
sage_show_khsq (stdout, "L11n449", sq1.domain (), sq1, sq2);
}
void
dump_sage ()
{
char buf[1000];
#if 0
for (unsigned i = 10; i >= 1; i --)
{
if (i <= 10)
{
for (unsigned j = 1; j <= rolfsen_crossing_knots (i); j += block_size)
{
load (knot_kh_sq, knot_desc (knot_desc::ROLFSEN, i, j));
}
}
...
}
#endif
for (unsigned i = 12; i >= 1; i --)
{
map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > knot_kh_sq;
#if 0
sprintf (buf, "K%d.sage", i);
FILE *fp = fopen (buf, "w");
if (fp == 0)
{
stderror ("fopen: %s", buf);
exit (EXIT_FAILURE);
}
for (unsigned j = 1; j <= htw_knots (i); j += block_size)
{
load (knot_kh_sq, knot_desc (knot_desc::HTW, i, j));
}
for (map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > >::const_iter k = knot_kh_sq; k; k ++)
{
sprintf (buf, "K%s", k.key ().name ().c_str ());
sage_show_khsq (fp, buf, k.val ().first.domain (), k.val ().first, k.val ().second);
}
fclose (fp);
#endif
#if 1
sprintf (buf, "L%d.sage", i);
FILE *fp = fopen (buf, "w");
if (fp == 0)
{
stderror ("fopen: %s", buf);
exit (EXIT_FAILURE);
}
if (i <= 13)
{
for (unsigned j = 1; j <= mt_links (i); j += block_size)
{
load (knot_kh_sq, knot_desc (knot_desc::MT, i, j));
}
}
if (i == 14)
{
for (unsigned j = 1; j <= mt_links (14, 0); j += block_size)
{
load (knot_kh_sq, knot_desc (knot_desc::MT, 14, mt_links (14, 1) + j));
}
}
for (map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > >::const_iter k = knot_kh_sq; k; k ++)
{
sprintf (buf, "K%s", k.key ().name ().c_str ());
sage_show_khsq (fp, buf, k.val ().first.domain (), k.val ().first, k.val ().second);
}
fclose (fp);
#endif
}
}
void
convert_knot_sq ()
{
FILE *fp = open_file ("L14.files", "r");
char buf[1000];
gzfile_writer w ("L14_sq.dat.gz");
w.write_unsigned (76516);
set<knot_desc> keys;
while (fgets (buf, 1000, fp))
{
char *p = strchr (buf, '\n');
if (p)
{
assert (p[1] == 0);
*p = 0;
}
map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > knot_sq;
load (knot_sq, buf);
for (map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > >::const_iter k = knot_sq; k; k ++)
{
write (w, k.key ());
write (w, k.val ());
}
}
// write (w, knot_sq);
}
#if 0
void
test_knot_sq ()
{
const char *knot_sq_files[] = {
"knot_sq/rolfsen_sq.dat.gz",
"knot_sq/Ksmall_sq.dat.gz",
"knot_sq/K11_sq.dat.gz",
"knot_sq/K12_sq.dat.gz",
"knot_sq/Lsmall_sq.dat.gz",
"knot_sq/L11_sq.dat.gz",
"knot_sq/L12_sq.dat.gz",
0,
};
for (unsigned i = 0; knot_sq_files[i] != 0; i ++)
{
gzfile_reader r (knot_sq_files[i]);
map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > knot_sq (r);
printf ("|knot_sq| = %d\n", knot_sq.card ());
}
}
pair<mod_map<Z2>, mod_map<Z2> >
compute_kh_sq (const knot_desc &desc)
{
abort ();
}
#endif
template<class R> pair<ptr<const module<R> >, mod_map<R> >
FU_complex (ptr<const module<R> > CF,
mod_map<R> dF,
mod_map<R> dU)
{
unsigned n = CF->dim ();
basedvector<grading, 1> hq (n * 2);
for (unsigned i = 1; i <= n; i ++)
hq[i] = hq[n + i] = CF->generator_grading (i);
ptr<const module<R> > CFU = new explicit_module<R> (2 * n,
basedvector<R, 1> (),
hq);
map_builder<R> b (CFU);
for (unsigned i = 1; i <= n; i ++)
{
for (linear_combination_const_iter<R> j = dF[i]; j; j ++)
{
b[i].muladd (j.val (), j.key ());
b[n + i].muladd (j.val (), n + j.key ());
}
for (linear_combination_const_iter<R> j = dU[i]; j; j ++)
{
b[i].muladd (j.val (), n + j.key ());
}
}
mod_map<R> dFU (b);
// assert (dFU.compose (dFU) == 0);
// dFU.check_sseq_graded ();
return pair<ptr<const module<R> >, mod_map<R> > (CFU, dFU);
}
void
compute_twistedU ()
{
#if 0
for (unsigned i = 1; i <= 11; i ++)
for (unsigned j = 1; j <= htw_knots (i, 0); j ++)
{
knot_diagram kd (htw_knot (i, 0, j));
...}
#endif
for (unsigned i = 1; i <= 10; i ++)
for (unsigned j = 1; j <= mt_links (i, 0); j ++)
{
knot_diagram kd (mt_link (i, 0, j));
kd.marked_edge = 1;
show (kd); newline ();
typedef spanning_tree_complex<Z2>::R R;
spanning_tree_complex<Z2> c (kd);
ptr<const module<R> > C = c.C;
mod_map<R> d2 = c.twisted_d2 ();
assert (d2.compose (d2) == 0);
mod_map<R> d2U = c.twisted_d2Un (1);
assert (d2.compose (d2U) == d2U.compose (d2));
chain_complex_simplifier<R> s (C, d2, 2);
assert (s.new_d == 0);
printf ("|s.new_C| = %d\n", s.new_C->dim ());
pair<ptr<const module<R> >, mod_map<R> > p = FU_complex (C, d2, d2U);
ptr<const module<R> > CFU = p.first;
mod_map<R> dFU = p.second;
chain_complex_simplifier<R> s2 (CFU, dFU, 2);
assert (s2.new_d == 0);
printf ("|s2.new_C| = %d\n", s2.new_C->dim ());
if (s2.new_C->dim () < 2 * s.new_C->dim ())
printf (" > EXAMPLE\n");
}
}
void
test_forgetful_ss ()
{
typedef fraction_field<polynomial<Z2> > R;
for (unsigned i = 1; i <= 11; i ++)
for (unsigned j = 1; j <= mt_links (i, 0); j ++)
{
knot_diagram kd (mt_link (i, 0, j));
unsigned n = kd.num_components ();
if (n < 2)
continue;
show (kd); newline ();
unionfind<1> u (kd.num_edges ());
for (unsigned i = 1; i <= kd.n_crossings; i ++)
{
u.join (kd.ept_edge (kd.crossings[i][1]),
kd.ept_edge (kd.crossings[i][3]));
u.join (kd.ept_edge (kd.crossings[i][2]),
kd.ept_edge (kd.crossings[i][4]));
}
printf ("%d components:\n", n);
map<unsigned, unsigned> root_comp;
unsigned t = 0;
for (unsigned i = 1; i <= kd.num_edges (); i ++)
{
if (u.find (i) == i)
{
++ t;
root_comp.push (i, t);
}
}
assert (t == n);
printf (" kd w: %d\n", kd.writhe ());
multivariate_laurentpoly<Z> disj_P = 1;
for (unsigned k = 1; k <= n; k ++)
{
knot_diagram comp (SUBLINK, smallbitset (n, unsigned_2pow (k - 1)), kd);
unsigned w = 0;
for (unsigned i = 1; i <= kd.n_crossings; i ++)
{
if (root_comp(u.find (kd.ept_edge (kd.crossings[i][1]))) == k
&& root_comp(u.find (kd.ept_edge (kd.crossings[i][2]))) == k)
{
if (kd.is_to_ept (kd.crossings[i][1]) == kd.is_to_ept (kd.crossings[i][4]))
w ++;
else
w --;
}
}
printf (" % 2d w: %d\n", k, w);
cube<R> c (comp);
mod_map<R> d = c.compute_d (1, 0, 0, 0, 0);
chain_complex_simplifier<R> s (c.khC, d, 1);
assert (s.new_d == 0);
multivariate_laurentpoly<Z> P = s.new_C->free_ell_poincare_polynomial ();
printf (" % 2d P: ", k);
display (P);
disj_P *= (P
* multivariate_laurentpoly<Z> (1, VARIABLE, 1, w)
);
}
cube<R> c (kd);
#if 0
for (unsigned i = 0; i < c.n_resolutions; i ++)
{
smallbitset state (kd.n_crossings, i);
smoothing s (kd, state);
s.show_self (kd, state);
newline ();
}
#endif
mod_map<R> untwisted_d = c.compute_d (1, 0, 0, 0, 0);
assert (untwisted_d.compose (untwisted_d) == 0);
chain_complex_simplifier<R> s1 (c.khC, untwisted_d, 1);
assert (s1.new_d == 0);
multivariate_laurentpoly<Z> P1 = s1.new_C->free_ell_poincare_polynomial ();
display (" link P : ", P1);
display (" disj_P (adj): ", disj_P);
mod_map<R> d = untwisted_d;
for (unsigned x = 1; x <= kd.n_crossings; x ++)
{
unsigned r1 = u.find (kd.ept_edge (kd.crossings[x][1])),
r2 = u.find (kd.ept_edge (kd.crossings[x][2]));
d = d + c.compute_dinv (x)*(R (polynomial<Z2> (1, root_comp(r1))) + 1);
d = d + c.compute_dinv (x)*(R (polynomial<Z2> (1, root_comp(r2))) + 1);
#if 0
R hbar (polynomial<Z2> (1, 1));
R hbarp1 = hbar + 1;
// if (u.find (kd.ept_edge (kd.crossings[x][1])) != u.find (kd.ept_edge (kd.crossings[x][2])))
if (u.find (kd.ept_edge (kd.crossings[x][1])) != comp1_root)
d = d + c.compute_dinv (x)*hbarp1;
if (u.find (kd.ept_edge (kd.crossings[x][2])) != comp1_root)
d = d + c.compute_dinv (x)*hbarp1;
#endif
}
#if 0
mod_map<R> h = d.compose (d);
display ("h:\n", h);
#endif
assert (d.compose (d) == 0);
// display ("d:\n", d);
chain_complex_simplifier<R> s2 (c.khC, d, -1);
assert (s2.new_d == 0);
multivariate_laurentpoly<Z> P2 = (s2.new_C->free_ell_poincare_polynomial ()
* multivariate_laurentpoly<Z> (1, VARIABLE, 1, kd.writhe ())
);
display (" Einf P (adj): ", P2);
if (disj_P == P2)
printf (" disj_P == Einf P (adj): YES!\n");
else
printf (" disj_P != Einf P (adj): NO :-(!\n");
}
}
/* Kh homotopy type:
- check for CP^2,
- mutation invariance
- K-theory */
map<grading, basedvector<int, 1> >
compute_st (mod_map<Z2> sq1,
mod_map<Z2> sq2)
{
ptr<const module<Z2> > H = sq1.domain ();
map<grading, basedvector<int, 1> > st;
bool first = 1;
set<grading> gs = H->gradings ();
for (set_const_iter<grading> gg = gs; gg; gg ++)
{
grading hq = gg.val (),
h1q (hq.h + 1, hq.q),
h2q (hq.h + 2, hq.q);
ptr<const free_submodule<Z2> > H_hq = H->graded_piece (hq),
H_h1q = H->graded_piece (h1q),
H_h2q = H->graded_piece (h2q);
mod_map<Z2> S = sq2.restrict (H_hq, H_h2q),
A = sq1.restrict (H_hq, H_h1q),
B = sq1.restrict (H_h1q, H_h2q);
ptr<const free_submodule<Z2> > S_im = S.image (),
A_ker = A.kernel (),
B_im = B.image ();
ptr<const free_submodule<Z2> > inter = S_im->intersection (B_im);
mod_map<Z2> S_res = S.restrict_from (A_ker);
ptr<const free_submodule<Z2> > S_res_im = S_res.image ();
ptr<const free_submodule<Z2> > res_inter = S_res_im->intersection (B_im);
int r1 = S_im->dim ();
int r2 = S_res_im->dim ();
int r3 = inter->dim ();
int r4 = res_inter->dim ();
int s1 = r2 - r4,
s2 = r1 - r2 - r3 + r4,
s3 = r4,
s4 = r3 - r4;
if (!s1 && !s2 && !s3 && !s4)
continue;
basedvector<int, 1> s (4);
s[1] = s1;
s[2] = s2;
s[3] = s3;
s[4] = s4;
st.push (hq, s);
}
return st;
}
void
test_knot_sq ()
{
const char *knot_sq_files[] = {
"knot_sq/rolfsen_sq.dat.gz",
"knot_sq/Ksmall_sq.dat.gz",
"knot_sq/K11_sq.dat.gz",
"knot_sq/K12_sq.dat.gz",
"knot_sq/K13_sq.dat.gz",
#if 0
"knot_sq/K14_sq.dat.gz",
"knot_sq/Lsmall_sq.dat.gz",
"knot_sq/L11_sq.dat.gz",
"knot_sq/L12_sq.dat.gz",
"knot_sq/L13_sq.dat.gz",
"knot_sq/L14_sq.dat.gz",
#endif
0,
};
hashset<knot_desc> mutants;
for (unsigned i = 11; i <= 15; i ++)
{
basedvector<basedvector<unsigned, 1>, 1> groups
= mutant_knot_groups (i);
for (unsigned j = 1; j <= groups.size (); j ++)
{
for (unsigned k = 1; k <= groups[j].size (); k ++)
mutants += knot_desc (knot_desc::HTW, i, groups[j][k]);
}
}
printf ("|mutants| = %d\n", mutants.card ());
basedvector<unsigned, 1> width_hist (7);
for (unsigned i = 1; i <= width_hist.size (); i ++)
width_hist[i] = 0;
hashmap<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > mutant_knot_sq,
mutant_mknot_sq;
{
gzfile_reader r ("knot_sq/mut_mknot.dat.gz");
mutant_mknot_sq = hashmap<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > (r);
for (hashmap<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > >::const_iter i = mutant_mknot_sq; i; i ++)
{
knot_desc desc = i.key ();
// careful: duplicate code!
pair<mod_map<Z2>, mod_map<Z2> > p = i.val ();
mod_map<Z2> sq1 = p.first,
sq2 = p.second;
ptr<const module<Z2> > H = sq1.domain ();
unsigned w = homological_width (H);
assert (w >= 1 && w <= 7);
width_hist[w] ++;
assert (sq1.compose (sq1) == 0);
assert (sq2.compose (sq2) == sq1.compose (sq2).compose (sq1));
map<grading, basedvector<int, 1> > st
= compute_st (sq1, sq2);
for (map_const_iter<grading, basedvector<int, 1> > i = st; i; i ++)
{
if (i.val ()[1] != 0)
{
show (desc); newline ();
printf (" > has CP^2\n");
}
}
}
}
for (unsigned i = 0; knot_sq_files[i]; i ++)
{
gzfile_reader r (knot_sq_files[i]);
printf ("loading %s...\n", knot_sq_files[i]);
unsigned n = r.read_unsigned ();
for (unsigned i = 1; i <= n; i ++)
{
knot_desc desc (r);
pair<mod_map<Z2>, mod_map<Z2> > p (r);
if (mutants % desc)
mutant_knot_sq.push (desc, p);
mod_map<Z2> sq1 = p.first,
sq2 = p.second;
ptr<const module<Z2> > H = sq1.domain ();
unsigned w = homological_width (H);
assert (w >= 1 && w <= 7);
width_hist[w] ++;
assert (sq1.compose (sq1) == 0);
assert (sq2.compose (sq2) == sq1.compose (sq2).compose (sq1));
map<grading, basedvector<int, 1> > st
= compute_st (sq1, sq2);
for (map_const_iter<grading, basedvector<int, 1> > i = st; i; i ++)
{
if (i.val ()[1] != 0)
{
show (desc); newline ();
printf (" > has CP^2\n");
}
}
}
}
printf ("|mutant_knot_sq| = %d\n", mutant_knot_sq.card ());
printf ("width_hist:\n");
for (unsigned i = 1; i <= width_hist.size (); i ++)
{
printf (" % 2d: %d\n", i, width_hist[i]);
}
unsigned missing = 0,
compared = 0;
for (unsigned i = 11; i <= 15; i ++)
{
basedvector<basedvector<unsigned, 1>, 1> groups
= mutant_knot_groups (i);
for (unsigned j = 1; j <= groups.size (); j ++)
{
bool first = 1;
knot_desc desc_1;
ptr<const module<Z2> > H_1;
map<grading, basedvector<int, 1> > st_1;
for (unsigned k = 1; k <= groups[j].size (); k ++)
{
knot_desc desc (knot_desc::HTW, i, groups[j][k]);
if (mutant_knot_sq % desc)
{
pair<mod_map<Z2>, mod_map<Z2> > p = mutant_knot_sq(desc);
mod_map<Z2> sq1 = p.first,
sq2 = p.second;
ptr<const module<Z2> > H = sq1.domain ();
map<grading, basedvector<int, 1> > st = compute_st (sq1, sq2);
if (first)
{
first = 0;
desc_1 = desc;
H_1 = H;
st_1 = st;
}
else
{
if (H_1->free_poincare_polynomial ()
== H->free_poincare_polynomial ())
{
if (st_1 != st)
{
printf ("> DIFFER\n");
display (" ", desc_1);
display (" ", desc);
}
compared ++;
}
else if (mutant_mknot_sq % desc)
{
pair<mod_map<Z2>, mod_map<Z2> > mp = mutant_mknot_sq(desc);
mod_map<Z2> msq1 = mp.first,
msq2 = mp.second;
ptr<const module<Z2> > mH = msq1.domain ();
assert (H_1->free_poincare_polynomial ()
== mH->free_poincare_polynomial ());
map<grading, basedvector<int, 1> > mst
= compute_st (msq1, msq2);
if (st_1 != mst)
{
printf ("> DIFFER\n");
display (" ", desc_1);
display (" ", desc);
}
compared ++;
}
else
missing ++;
}
}
else
missing ++;
}
}
}
printf ("missing = %d\n", missing);
printf ("compared = %d\n", compared);
}
void
compute_mutant_mirrors ()
{
const char *knot_sq_files[] = {
// "knot_sq/K11_sq.dat.gz",
// "knot_sq/K12_sq.dat.gz",
// "knot_sq/K13_sq.dat.gz",
"knot_sq/K14_sq.dat.gz",
0,
};
unsigned minn = 14,
maxn = 14;
hashset<knot_desc> mutants;
for (unsigned i = minn; i <= maxn; i ++)
{
basedvector<basedvector<unsigned, 1>, 1> groups
= mutant_knot_groups (i);
for (unsigned j = 1; j <= groups.size (); j ++)
{
for (unsigned k = 1; k <= groups[j].size (); k ++)
mutants += knot_desc (knot_desc::HTW, i, groups[j][k]);
}
}
printf ("|mutants| = %d\n", mutants.card ());
hashmap<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > mutant_knot_sq;
for (unsigned i = 0; knot_sq_files[i]; i ++)
{
gzfile_reader r (knot_sq_files[i]);
printf ("loading %s...\n", knot_sq_files[i]);
unsigned n = r.read_unsigned ();
for (unsigned i = 1; i <= n; i ++)
{
knot_desc k (r);
pair<mod_map<Z2>, mod_map<Z2> > p (r);
if (mutants % k)
mutant_knot_sq.push (k, p);
}
}
printf ("|mutant_knot_sq| = %d\n", mutant_knot_sq.card ());
map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > mknot_kh_sq;
for (unsigned i = minn; i <= maxn; i ++)
{
basedvector<basedvector<unsigned, 1>, 1> groups
= mutant_knot_groups (i);
for (unsigned j = 1; j <= groups.size (); j ++)
{
unsigned n = groups[j].size ();
basedvector<ptr<const module<Z2> >, 1> group_H (n);
for (unsigned k = 1; k <= n; k ++)
{
knot_desc desc (knot_desc::HTW, i, groups[j][k]);
pair<mod_map<Z2>, mod_map<Z2> > p = mutant_knot_sq(desc);
group_H[k] = p.first.domain ();
}
for (unsigned k = 2; k <= n; k ++)
{
if (group_H[1]->free_poincare_polynomial ()
!= group_H[k]->free_poincare_polynomial ())
{
knot_desc desc (knot_desc::HTW, i, groups[j][k]);
show (desc); newline ();
knot_diagram kd (MIRROR, desc.diagram ());
cube<Z2> c (kd);
mod_map<Z2> d = c.compute_d (1, 0, 0, 0, 0);
chain_complex_simplifier<Z2> s (c.khC, d, 1);
assert (s.new_d == 0);
steenrod_square sq (c, d, s);
mod_map<Z2> sq1 = sq.sq1 ();
mod_map<Z2> sq2 = sq.sq2 ();
ptr<const module<Z2> > H = sq1.domain ();
assert (group_H[1]->free_poincare_polynomial ()
== H->free_poincare_polynomial ());
char buf[1000];
assert (desc.t == knot_desc::HTW);
sprintf (buf, "mknot/K%d_%d.dat.gz", desc.i, desc.j);
gzfile_writer w (buf);
write (w, desc);
write (w, sq1);
write (w, sq2);
}
}
}
}
}
void
convert_mknots ()
{
hashmap<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > mutant_mknot_sq;
for (unsigned i = 11; i <= 15; i ++)
{
basedvector<basedvector<unsigned, 1>, 1> groups
= mutant_knot_groups (i);
for (unsigned j = 1; j <= groups.size (); j ++)
{
unsigned n = groups[j].size ();
for (unsigned k = 1; k <= n; k ++)
{
knot_desc desc (knot_desc::HTW, i, groups[j][k]);
char buf[1000];
assert (desc.t == knot_desc::HTW);
sprintf (buf, "mknot/K%d_%d.dat.gz", desc.i, desc.j);
struct stat stat_buf;
if (stat (buf, &stat_buf) != 0)
{
if (errno != ENOENT)
{
stderror ("stat: %s", buf);
exit (EXIT_FAILURE);
}
}
else
{
printf ("loading %s...\n", buf);
gzfile_reader r (buf);
knot_desc desc_2 (r);
assert (desc_2 == desc);
mod_map<Z2> sq1 (r);
mod_map<Z2> sq2 (r);
mutant_mknot_sq.push (desc,
pair<mod_map<Z2>, mod_map<Z2> > (sq1, sq2));
}
}
}
}
gzfile_writer w ("knot_sq/mut_mknot.dat.gz");
write (w, mutant_mknot_sq);
}
int
main ()
{
compute_mutant_mirrors ();
return 0;
test_knot_sq ();
convert_mknots ();
test_forgetful_ss ();
compute_twistedU ();
#if 1
knot_desc desc;
desc.t = knot_desc::HTW;
desc.i = 15;
desc.j = 15020;
char buf[1000];
sprintf (buf, "/scratch/network/cseed/incoming/K%d_%d.dat.gz",
desc.i, desc.j);
desc = knot_desc (knot_desc::ROLFSEN, 3, 1);
knot_diagram kd = desc.diagram ();
cube<Z2> c (kd);
mod_map<Z2> d = c.compute_d (1, 0, 0, 0, 0);
chain_complex_simplifier<Z2> s (c.khC, d, 1);
assert (s.new_d == 0);
steenrod_square sq (c, d, s);
mod_map<Z2> sq1 = sq.sq1 ();
mod_map<Z2> sq2 = sq.sq2 ();
assert (sq1.compose (sq1) == 0);
assert (sq2.compose (sq2) + sq1.compose (sq2).compose (sq1) == 0);
#if 1
{
file_writer w (buf);
write (w, sq1);
write (w, sq2);
}
#endif
#endif
return 0;
// convert_knot_sq ();
test_knot_sq ();
// test_sage_show ();
dump_sage ();
#if 0
knot_diagram kd (rolfsen_knot (8, 19));
show (kd); newline ();
cube<Z2> c (kd);
mod_map<Z2> d = c.compute_d (1, 0, 0, 0, 0);
chain_complex_simplifier<Z2> s (c.khC, d, 1);
steenrod_square sq (c, d, s);
mod_map<Z2> sq1 = sq.sq1 ();
mod_map<Z2> sq2 = sq.sq2 ();
assert (sq1.compose (sq1) == 0);
assert (sq2.compose (sq2) + sq1.compose (sq2).compose (sq1) == 0);
display ("sq1:\n", sq1);
display ("sq2:\n", sq2);
{
file_writer w ("sqtest.dat");
write (w, sq1);
write (w, sq2);
}
{
file_reader r ("sqtest.dat");
mod_map<Z2> sq1p (r);
mod_map<Z2> sq2p (r);
display ("sq1p:\n", sq1p);
display ("sq2p:\n", sq2p);
}
#endif
#if 0
knot_diagram kd (mt_link (10, 0, 9));
cube<Z2> c (kd);
#endif
#if 0
for (unsigned i = 1; i <= 10; i ++)
for (unsigned j = 1; j <= mt_links (i, 0); j ++)
{
knot_diagram kd (mt_link (i, 0, j));
kd.marked_edge = 1;
cube<Z2> c (kd, 1);
sseq ss = compute_szabo_sseq (c);
ss.texshow (stdout, kd.name);
}
#endif
#if 0
#if 1
for (unsigned i = 10; i <= 10; i ++)
for (unsigned j = 124; j <= rolfsen_crossing_knots (i); j ++)
{
knot_diagram kd (rolfsen_knot (i, j));
#endif
#if 0
for (unsigned i = 1; i <= 10; i ++)
for (unsigned j = 1; j <= mt_links (i, 0); j ++)
{
#endif
#if 0
for (unsigned i = 11; i <= 11; i ++)
for (unsigned j = 1; j <= htw_knots (i, 0); j ++)
{
#endif
// knot_diagram kd (htw_knot (i, 0, j));
// knot_diagram kd (mt_link (i, 0, j));
kd.marked_edge = 1;
show (kd); newline ();
cube<Z2> c (kd, 1);
#if 0
mod_map<Z2> d1 = c.compute_d (1, 0, 0, 0, 0);
chain_complex_simplifier<Z2> s (c.khC, d1, 1);
assert (s.new_d == 0);
#endif
sseq ss = compute_szabo_sseq (c);
multivariate_laurentpoly<Z> Phat =
ss.pages[ss.pages.size ()].delta_poincare_polynomial (ss.bounds);
typedef spanning_tree_complex<Z2>::R R;
spanning_tree_complex<Z2> spanc (kd);
mod_map<R> d2 = spanc.twisted_d2 ();
mod_map<R> d2U = spanc.twisted_d2Un (1);
chain_complex_simplifier<R> s2 (spanc.C, d2, 2);
assert (s2.new_d == 0);
mod_map<R> H_d2U = s2.pi.compose (d2U).compose (s2.iota);
assert (H_d2U.compose (H_d2U) == 0);
ptr<const module<R> > ker = H_d2U.kernel ();
ptr<const module<R> > quot = s2.new_C->quotient (H_d2U.image ());
multivariate_laurentpoly<Z> Pminus1
= ker->free_delta_poincare_polynomial (),
PminusU = quot->free_delta_poincare_polynomial ();
if (PminusU != Pminus1)
{
display (" HFhat: ", Phat);
// display (" HF-: ", Pminus);
display (" HF- (1): ", Pminus1);
display (" HF- (U): ", PminusU);
}
}
#endif
#if 0
hwidth_knots = basedvector<unsigned, 1> (10);
for (unsigned i = 1; i <= hwidth_knots.size (); i ++)
hwidth_knots[i] = 0;
map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > > knot_kh_sq;
for (unsigned i = 14; i >= 1; i --)
{
#if 0
if (i <= 10)
{
for (unsigned j = 1; j <= rolfsen_crossing_knots (i); j += block_size)
{
load (knot_kh_sq, knot_desc (knot_desc::ROLFSEN, i, j));
}
}
#endif
for (unsigned j = 1; j <= htw_knots (i); j += block_size)
{
load (knot_kh_sq, knot_desc (knot_desc::HTW, i, j));
}
if (i <= 13)
{
for (unsigned j = 1; j <= mt_links (i); j += block_size)
{
load (knot_kh_sq, knot_desc (knot_desc::MT, i, j));
}
}
if (i == 14)
{
for (unsigned j = 1; j <= mt_links (14, 0); j += block_size)
{
load (knot_kh_sq, knot_desc (knot_desc::MT, 14, mt_links (14, 1) + j));
}
}
}
unsigned total_knots = 0;
printf ("hwidth_knots:\n");
for (unsigned i = 1; i <= hwidth_knots.size (); i ++)
{
printf (" % 2d: %d\n", i, hwidth_knots[i]);
total_knots += hwidth_knots[i];
}
printf ("total_knots = %d\n", total_knots);
printf ("|knot_kh_sq| = %d\n", knot_kh_sq.card ());
#if 0
map<pair<multivariate_laurentpoly<Z>,
map<grading, unsigned> >,
pair<knot_desc,
map<grading, basedvector<int, 1> > > > P_sq1_knot_st;
set<multivariate_laurentpoly<Z> > Ps;
basedvector<unsigned, 1> collisons (10);
for (unsigned i = 1; i <= 10; i ++)
collisons[i] = 0;
for (map<knot_desc,
pair<mod_map<Z2>, mod_map<Z2> > >::const_iter i = knot_kh_sq; i; i ++)
{
show_st (knot_kh_sq, i.key ());
mod_map<Z2> sq1 = i.val ().first;
mod_map<Z2> sq2 = i.val ().second;
#if 0
display ("sq1:\n", sq1);
display ("sq2:\n", sq2);
#endif
printf ("%s ", i.key ().name ().c_str ());
assert (sq1.compose (sq1) == 0);
assert (sq2.compose (sq2) + sq1.compose (sq2).compose (sq1) == 0);
ptr<const module<Z2> > H = sq1.domain ();
unsigned hwidth = homological_width (H);
map<grading, basedvector<int, 1> > st;
map<grading, unsigned> sq1_ranks;
bool first = 1;
set<grading> gs = H->gradings ();
for (set_const_iter<grading> gg = gs; gg; gg ++)
{
grading hq = gg.val (),
h1q (hq.h + 1, hq.q),
h2q (hq.h + 2, hq.q),
h3q (hq.h + 3, hq.q);
// printf ("(%d, %d):\n", hq.h, hq.q);
ptr<const free_submodule<Z2> > H_hq = H->graded_piece (hq),
H_h1q = H->graded_piece (h1q),
H_h2q = H->graded_piece (h2q),
H_h3q = H->graded_piece (h3q);
mod_map<Z2> S = sq2.restrict (H_hq, H_h2q),
T = sq2.restrict (H_h1q, H_h3q),
A = sq1.restrict (H_hq, H_h1q),
B = sq1.restrict (H_h1q, H_h2q),
C = sq1.restrict (H_h2q, H_h3q);
ptr<const free_submodule<Z2> > Sker = S.kernel (),
Sim = S.image (),
Tker = T.kernel (),
Tim = T.image (),
Aker = A.kernel (),
Aim = A.image (),
Bker = B.kernel (),
Bim = B.image (),
Cker = C.kernel (),
Cim = C.image ();
sq1_ranks.push (hq, Aim->dim ());
mod_map<Z2> ArSker = A.restrict_from (Sker);
mod_map<Z2> SrAker = S.restrict_from (Aker);
mod_map<Z2> TrAim = T.restrict_from (Aim);
mod_map<Z2> TrBker = T.restrict_from (Bker);
mod_map<Z2> BrTker = B.restrict_from (Tker);
mod_map<Z2> CrSim = C.restrict_from (Sim);
ptr<const free_submodule<Z2> > ArSker_im = ArSker.image ();
ptr<const free_submodule<Z2> > SrAker_im = SrAker.image ();
ptr<const free_submodule<Z2> > TrAim_im = TrAim.image ();
ptr<const free_submodule<Z2> > TrBker_im = TrBker.image ();
ptr<const free_submodule<Z2> > BrTker_im = BrTker.image ();
ptr<const free_submodule<Z2> > CrSim_im = CrSim.image ();
mod_map<Z2> CrSrAker_im = C.restrict_from (SrAker_im);
mod_map<Z2> TrArSker_im = T.restrict_from (ArSker_im);
ptr<const free_submodule<Z2> > CrSrAker_im_im = CrSrAker_im.image ();
ptr<const free_submodule<Z2> > TrArSker_im_im = TrArSker_im.image ();
ptr<const free_submodule<Z2> > Aker_cap_Sker = Aker->intersection (Sker);
ptr<const free_submodule<Z2> > Aim_cap_Tker = Aim->intersection (Tker);
ptr<const free_submodule<Z2> > Bker_cap_Tker = Bker->intersection (Tker);
ptr<const free_submodule<Z2> > ArSker_im_cap_Tker = ArSker_im->intersection (Tker);
ptr<const free_submodule<Z2> > Sim_cap_Bim = Sim->intersection (Bim);
ptr<const free_submodule<Z2> > Sim_cap_BrTker_im = Sim->intersection (BrTker_im);
ptr<const free_submodule<Z2> > Sim_cap_Cker = Sim->intersection (Cker);
ptr<const free_submodule<Z2> > SrAker_im_cap_Bim = SrAker_im->intersection (Bim);
ptr<const free_submodule<Z2> > SrAker_im_cap_BrTker_im = SrAker_im->intersection (BrTker_im);
ptr<const free_submodule<Z2> > SrAker_im_cap_Cker = SrAker_im->intersection (Cker);
ptr<const free_submodule<Z2> > Tim_cap_Cim = Tim->intersection (Cim);
ptr<const free_submodule<Z2> > TrAim_im_cap_Cim = TrAim_im->intersection (Cim);
ptr<const free_submodule<Z2> > TrBker_im_cap_Cim = TrBker_im->intersection (Cim);
ptr<const free_submodule<Z2> > TrArSker_im_im_cap_Cim = TrBker_im->intersection (Cim);
ptr<const free_submodule<Z2> > Tim_cap_CrSim_im = Tim->intersection (CrSim_im);
ptr<const free_submodule<Z2> > TrAim_im_cap_CrSim_im = TrAim_im->intersection (CrSim_im);
ptr<const free_submodule<Z2> > TrBker_im_cap_CrSim_im = TrBker_im->intersection (CrSim_im);
ptr<const free_submodule<Z2> > TrArSker_im_im_cap_CrSim_im = TrBker_im->intersection (CrSim_im);
ptr<const free_submodule<Z2> > Tim_cap_CrSrAker_im_im = Tim->intersection (CrSrAker_im_im);
ptr<const free_submodule<Z2> > TrAim_im_cap_CrSrAker_im_im = TrAim_im->intersection (CrSrAker_im_im);
ptr<const free_submodule<Z2> > TrBker_im_cap_CrSrAker_im_im = TrBker_im->intersection (CrSrAker_im_im);
ptr<const free_submodule<Z2> > TrArSker_im_im_cap_CrSrAker_im_im = TrBker_im->intersection (CrSrAker_im_im);
basedvector<int, 1> v;
v.append (Sker->dim ());
v.append (Sim->dim ());
v.append (Tker->dim ());
v.append (Tim->dim ());
v.append (Aker->dim ());
v.append (Aim->dim ());
v.append (Bker->dim ());
v.append (Bim->dim ());
v.append (Cker->dim ());
v.append (Cim->dim ());
v.append (ArSker_im->dim ());
v.append (SrAker_im->dim ());
v.append (TrAim_im->dim ());
v.append (TrBker_im->dim ());
v.append (BrTker_im->dim ());
v.append (CrSim_im->dim ());
v.append (CrSrAker_im_im->dim ());
v.append (TrArSker_im_im->dim ());
v.append (Aker_cap_Sker->dim ());
v.append (Aim_cap_Tker->dim ());
v.append (Bker_cap_Tker->dim ());
v.append (ArSker_im_cap_Tker->dim ());
v.append (Sim_cap_Bim->dim ());
v.append (Sim_cap_BrTker_im->dim ());
v.append (Sim_cap_Cker->dim ());
v.append (SrAker_im_cap_Bim->dim ());
v.append (SrAker_im_cap_BrTker_im->dim ());
v.append (SrAker_im_cap_Cker->dim ());
v.append (Tim_cap_Cim->dim ());
v.append (TrAim_im_cap_Cim->dim ());
v.append (TrBker_im_cap_Cim->dim ());
v.append (TrArSker_im_im_cap_Cim->dim ());
v.append (Tim_cap_CrSim_im->dim ());
v.append (TrAim_im_cap_CrSim_im->dim ());
v.append (TrBker_im_cap_CrSim_im->dim ());
v.append (TrArSker_im_im_cap_CrSim_im->dim ());
v.append (Tim_cap_CrSim_im->dim ());
v.append (TrAim_im_cap_CrSim_im->dim ());
v.append (TrBker_im_cap_CrSim_im->dim ());
v.append (TrArSker_im_im_cap_CrSim_im->dim ());
st.push (hq, v);
}
newline ();
multivariate_laurentpoly<Z> P = H->free_poincare_polynomial ();
pair<pair<knot_desc,
map<grading, basedvector<int, 1> > > &,
bool> p = P_sq1_knot_st.find (pair<multivariate_laurentpoly<Z>,
map<grading, unsigned> > (P, sq1_ranks));
if (p.second)
{
collisons[hwidth] ++;
Ps += P;
if (p.first.second != st)
{
printf ("DIFFER:\n");
printf ("hwidth = %d\n", hwidth);
show_st (knot_kh_sq, p.first.first);
show_st (knot_kh_sq, i.key ());
printf ("Kh[");
planar_diagram (p.first.first.diagram ()).show_knottheory ();
printf (", Modulus -> Null][q,t] === Kh[");
planar_diagram (i.key ().diagram ()).show_knottheory ();
printf (", Modulus -> Null][q,t]\n");
#if 0
printf ("%s:\n",
p.first.first.name ().c_str ());
for (map<grading, basedvector<int, 1> >::const_iter j = p.first.second; j; j ++)
{
printf (" (%d, %d) -> [",
j.key ().h, j.key ().q);
for (unsigned k = 1; k <= j.val ().size (); k ++)
{
if (k > 1)
printf (",");
printf ("%d", j.val ()[k]);
}
newline ();
}
printf ("%s:\n",
i.key ().name ().c_str ());
for (map<grading, basedvector<int, 1> >::const_iter j = st; j; j ++)
{
printf (" (%d, %d) -> [",
j.key ().h, j.key ().q);
for (unsigned k = 1; k <= j.val ().size (); k ++)
{
if (k > 1)
printf (",");
printf ("%d", j.val ()[k]);
}
newline ();
}
#endif
}
}
else
{
p.first.first = i.key ();
p.first.second = st;
}
}
printf ("groups = %d\n", Ps.card ());
printf ("collisons:\n");
for (unsigned i = 1; i <= 10; i ++)
printf (" % 2d: %d\n", i, collisons[i]);
#endif
#endif
#if 0
#if 0
reader r ("sqtest.dat");
knot_desc kd (knot_desc::ROLFSEN, 8, 19);
mod_map<Z2> sq1 (r);
mod_map<Z2> sq2 (r);
ptr<const module<Z2> > H = sq1.domain ();
display ("sq1:\n", sq1);
display ("sq2:\n", sq2);
#endif
for (unsigned i = 1; i <= 10; i ++)
for (unsigned j = 1; j <= rolfsen_crossing_knots (i); j ++)
{
#if 0
// (Knot Atlas) L10n18
int xings[10][4] = {
{ 6, 1, 7, 2 },
{ 18, 7, 19, 8 },
{ 4,19,1,20 },
{ 5,12,6,13 },
{ 8,4,9,3 },
{ 13,17,14,16 },
{ 9,15,10,14 },
{ 15,11,16,10 },
{ 11,20,12,5 },
{ 2,18,3,17 },
};
knot_diagram kd (planar_diagram ("L10n18", 10, xings));
#endif
// (Knot Atlas) L10n102
int xings[10][4] = {
{ 6,1,7,2 }, { 10,3,11,4 }, { 14,7,15,8 }, { 8,13,5,14 }, { 11,18,12,19 }, { 15,20,16,17 }, { 19,16,20,9 }, { 17,12,18,13 }, { 2,5,3,6 }, { 4,9,1,10 },
};
knot_diagram kd (planar_diagram ("L10n102", 10, xings));
// knot_diagram kd (rolfsen_knot (i, j));
// knot_diagram kd (mt_link (10, 0, 18));
// show (kd); newline ();
printf ("%s ", kd.name.c_str ());
cube<Z2> c (kd);
mod_map<Z2> d = c.compute_d (1, 0, 0, 0, 0);
chain_complex_simplifier<Z2> s (c.khC, d, 1);
steenrod_square sq (c, d, s);
mod_map<Z2> sq1 = sq.sq1 ();
mod_map<Z2> sq2 = sq.sq2 ();
assert (sq1.compose (sq1) == 0);
assert (sq2.compose (sq2) + sq1.compose (sq2).compose (sq1) == 0);
ptr<const module<Z2> > H = sq1.domain ();
bool first = 1;
// ???
set<grading> gs = H->gradings ();
for (set_const_iter<grading> i = gs; i; i ++)
{
grading hq = i.val (),
h1q (hq.h + 1, hq.q),
h2q (hq.h + 2, hq.q);
// printf ("(%d, %d):\n", hq.h, hq.q);
ptr<const free_submodule<Z2> > H_hq = H->graded_piece (hq),
H_h1q = H->graded_piece (h1q),
H_h2q = H->graded_piece (h2q);
mod_map<Z2> whole = sq2.restrict (H_hq, H_h2q),
tail = sq1.restrict (H_hq, H_h1q),
head = sq1.restrict (H_h1q, H_h2q);
ptr<const free_submodule<Z2> > whole_im = whole.image (),
tail_ker = tail.kernel (),
head_im = head.image ();
ptr<const free_submodule<Z2> > inter = whole_im->intersection (head_im);
mod_map<Z2> whole_res = whole.restrict_from (tail_ker);
ptr<const free_submodule<Z2> > whole_res_im = whole_res.image ();
ptr<const free_submodule<Z2> > res_inter = whole_res_im->intersection (head_im);
int r1 = whole_im->dim ();
int r2 = whole_res_im->dim ();
int r3 = inter->dim ();
int r4 = res_inter->dim ();
if (r1 == 0
&& r2 == 0
&& r3 == 0
&& r4 == 0)
continue;
// printf (" r = (%d, %d, %d, %d)\n", r1, r2, r3, r4);
int s1 = r2 - r4,
s2 = r1 - r2 - r3 + r4,
s3 = r4,
s4 = r3 - r4;
if (first)
{
#if 0
show (kd);
printf (": ");
#endif
first = 0;
}
else
printf (", ");
printf ("(%d, %d) -> (%d, %d, %d, %d)",
hq.h, hq.q,
s1, s2, s3, s4);
}
// if (!first)
newline ();
}
#endif
#if 0
// knot_diagram kd (htw_knot (12, 0, 48));
knot_diagram kd (htw_knot (10, 1, 23));
show (kd); newline ();
#endif
#if 0
for (int a = 1; a >= 0; a --)
for (unsigned i = 1; i <= 9; i ++)
for (unsigned j = 1; j <= htw_knots (i, a); j ++)
{
knot_diagram kd (htw_knot (i, a, j));
show (kd); newline ();
cube<Z2> c (kd);
mod_map<Z2> d = c.compute_d (1, 0, 0, 0, 0);
chain_complex_simplifier<Z2> s (c.khC, d, 1);
steenrod_square sq (c, d, s);
mod_map<Z2> sq1 = sq.sq1 ();
// display ("sq1:\n", sq1);
mod_map<Z2> sq2 = sq.sq2 ();
if (a)
assert (sq2 == 0);
else
display ("sq2:\n", sq2);
assert (sq1.compose (sq1) == 0);
assert (sq2.compose (sq2) + sq1.compose (sq2).compose (sq1) == 0);
}
#endif
#if 0
typedef Z2 R;
for (unsigned i = 12; i <= 12; i ++)
for (unsigned j = 1; j <= htw_knots (i, 0); j ++)
{
knot_diagram kd (htw_knot (i, 0, j));
show (kd); newline ();
cube<R> c (kd);
mod_map<R> d = c.compute_d (1, 0, 0, 0, 0);
chain_complex_simplifier<R> s (c.khC, d, 1);
// assert (s.new_d == 0);
printf ("|s.new_C| = %d\n", s.new_C->dim ());
}
#endif
#if 0
map<multivariate_laurentpoly<Z>,
set<triple<unsigned, int, unsigned> > > kh_knot_map;
for (int a = 1; a >= 0; a --)
for (unsigned i = 1; i <= 12; i ++)
for (unsigned j = 1; j <= htw_knots (i, a); j ++)
{
knot_diagram kd (htw_knot (i, a, j));
kd.marked_edge = 1;
show (kd); newline ();
cube<Z2> c (kd, 1);
mod_map<Z2> d = c.compute_d (1, 0, 0, 0, 0);
sseq_builder b (c.khC, d);
sseq ss = b.build_sseq ();
multivariate_laurentpoly<Z> P = ss.pages[1].poincare_polynomial (ss.bounds);
kh_knot_map[P].push (triple<unsigned, int, unsigned> (i, a, j));
}
{
writer w ("kh_knot_map.dat");
write (w, kh_knot_map);
}
#endif
#if 0
compute_show_kh_sq (knot_desc (knot_desc::ROLFSEN, 8, 19));
map<knot_desc,
triple<multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z> > > knot_kh_sq;
for (unsigned i = 1; i <= 10; i ++)
for (unsigned j = 1; j <= rolfsen_crossing_knots (i); j += 4000)
{
load (knot_kh_sq, knot_desc (knot_desc::ROLFSEN, i, j));
}
for (unsigned i = 1; i <= 12; i ++)
for (unsigned j = 1; j <= htw_knots (i); j += 4000)
{
load (knot_kh_sq, knot_desc (knot_desc::HTW, i, j));
}
for (unsigned i = 1; i <= 13; i ++)
for (unsigned j = 1; j <= mt_links (i); j += 4000)
{
load (knot_kh_sq, knot_desc (knot_desc::MT, i, j));
}
map<multivariate_laurentpoly<Z>,
triple<knot_desc,
multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z> > > m;
multivariate_laurentpoly<Z> groups;
for (map<knot_desc,
triple<multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z> > >::const_iter i = knot_kh_sq; i; i ++)
{
multivariate_laurentpoly<Z> P = i.val ().first,
P_sq1 = i.val ().second,
P_sq2 = i.val ().third;
pair<triple<knot_desc,
multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z> > &,
bool> p = m.find (P);
if (p.second)
{
groups += P;
if (p.first.second != P_sq1
|| p.first.third != P_sq2)
{
printf ("DIFFER ");
display (P);
printf (" %s\n", p.first.first.name ().c_str ());
printf (" sq1 "); display (p.first.second);
printf (" sq2 "); display (p.first.third);
printf (" %s\n", i.key ().name ().c_str ());
printf (" sq1 "); display (P_sq1);
printf (" sq2 "); display (P_sq2);
}
}
else
{
p.first.first = i.key ();
p.first.second = P_sq1;
p.first.third = P_sq2;
}
}
printf ("|groups| = %d\n", groups.card ());
printf ("done.\n");
#endif
#if 0
reader r ("kh_knot_map.dat");
map<multivariate_laurentpoly<Z>,
set<triple<unsigned, int, unsigned> > > kh_knot_map (r);
for (map<multivariate_laurentpoly<Z>,
set<triple<unsigned, int, unsigned> > >::const_iter i = kh_knot_map; i; i ++)
{
if (i.val ().card () == 1)
continue;
printf ("group\n");
bool first = 1;
multivariate_laurentpoly<Z> P, sq1_P, sq2_P;
for (set_const_iter<triple<unsigned, int, unsigned> > j = i.val (); j; j ++)
{
knot_diagram kd (htw_knot (j.val ().first,
j.val ().second,
j.val ().third));
printf (" "); show (kd); newline ();
triple<multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z>,
multivariate_laurentpoly<Z> > t = square (kd);
#if 0
display ("t.first = ", t.first);
display ("i.key () = ", i.key ());
assert (t.first == i.key ());
#endif
if (first)
{
P = t.first;
sq1_P = t.second;
sq2_P = t.third;
first = 0;
}
else
{
assert (P == t.first);
if (sq1_P != t.second)
printf (" prev sq1_P != sq1_P\n");
if (sq2_P != t.third)
printf (" prev sq2_P != sq2_P\n");
}
}
}
#endif
#if 0
typedef Z2 F;
typedef fraction_field<polynomial<F> > R;
for (unsigned i = 1; i <= 10; i ++)
for (unsigned j = 1; j <= htw_knots (i, 0); j ++)
{
knot_diagram kd (htw_knot (i, 0, j));
kd.marked_edge = 1;
show (kd); newline ();
spanning_tree_complex<F> st (kd);
mod_map<R> d2 = st.twisted_d2 ();
assert (d2.compose (d2) == 0);
mod_map<R> d2U1 = st.twisted_d2Un (1);
// mod_map<R> d2U1 = st.twisted_d2U1_test ();
assert (d2.compose (d2U1) + d2U1.compose (d2) == 0);
mod_map<R> d2U2 = st.twisted_d2Un (2);
assert (d2.compose (d2U2) + d2U2.compose (d2) + d2U1.compose (d2U1) == 0);
mod_map<R> d2U3 = st.twisted_d2Un (3);
assert (d2.compose (d2U3) + d2U3.compose (d2)
+ d2U2.compose (d2U1) + d2U1.compose (d2U2) == 0);
mod_map<R> d2U4 = st.twisted_d2Un (4);
assert (d2.compose (d2U4) + d2U4.compose (d2)
+ d2U3.compose (d2U1) + d2U1.compose (d2U3)
+ d2U2.compose (d2U2) == 0);
mod_map<R> d2U5 = st.twisted_d2Un (5);
assert (d2.compose (d2U5) + d2U5.compose (d2)
+ d2U4.compose (d2U1) + d2U1.compose (d2U4)
+ d2U3.compose (d2U2) + d2U2.compose (d2U3) == 0);
mod_map<R> d2U6 = st.twisted_d2Un (6);
assert (d2.compose (d2U6) + d2U6.compose (d2)
+ d2U5.compose (d2U1) + d2U1.compose (d2U5)
+ d2U4.compose (d2U2) + d2U2.compose (d2U4)
+ d2U3.compose (d2U3) == 0);
}
#endif
}