mirror of
				https://github.com/zenorogue/hyperrogue.git
				synced 2025-10-24 18:37:39 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			539 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			539 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| namespace reps {
 | |
| 
 | |
| constexpr int test_dim = 3;
 | |
| constexpr bool in_hyperbolic = true;
 | |
| 
 | |
| int edges, valence;
 | |
| 
 | |
| void prepare_tests() {
 | |
|   hr::start_game();
 | |
|   if(MDIM != test_dim) throw hr::hr_exception("fix your dimension");
 | |
|   if(!(in_hyperbolic ? hyperbolic : sphere)) throw hr::hr_exception("fix your geometry");
 | |
|   if(hr::variation != hr::eVariation::pure) throw hr::hr_exception("fix your variation"); 
 | |
|   if(quotient) throw hr::hr_exception("fix your quotient");
 | |
|   if(test_dim == 4) {
 | |
|     if(cginf.tiling_name != "{4,3,5}") throw hr::hr_exception("only {4,3,5} implemented in 3D");
 | |
|     edges = 4;
 | |
|     valence = 5;
 | |
|     }
 | |
|   else {
 | |
|     edges = hr::cwt.at->type;
 | |
|     bool ok;
 | |
|     valence = hr::get_valence(hr::cwt.at, 1, ok);
 | |
|     }
 | |
|   }
 | |
| 
 | |
| struct data {
 | |
|   using Number = hr::ld;
 | |
|   static constexpr int Dim = test_dim;
 | |
|   static constexpr int Flipped = in_hyperbolic ? test_dim-1 : -1;
 | |
|   };
 | |
| 
 | |
| struct countdata {
 | |
|   using Number = countfloat;
 | |
|   static constexpr int Dim = data::Dim;
 | |
|   static constexpr int Flipped = data::Flipped;
 | |
|   };
 | |
| 
 | |
| struct bigdata {
 | |
|   using Number = big;
 | |
|   static constexpr int Dim = data::Dim;
 | |
|   static constexpr int Flipped = data::Flipped;
 | |
|   };
 | |
| 
 | |
| using good = rep_linear_nn<bigdata>;
 | |
| 
 | |
| int debug; // 0 -- never, 1 -- only errors, 2 -- always
 | |
| 
 | |
| vector<cell*> randomwalk(std::mt19937& gen, cell *from, int dist) {
 | |
|   vector<cell*> res = { from };
 | |
|   while(celldistance(from, res.back()) < dist) {
 | |
|     int i = gen() % res.back()->type;
 | |
|     res.push_back(res.back()->cmove(i));
 | |
|     }
 | |
|   return res;
 | |
|   }
 | |
| 
 | |
| template<class N> N rand01(std::mt19937& gen) { return N(((gen() & HRANDMAX) | 1) / (HRANDMAX+1.)); }
 | |
| 
 | |
| vector<cell*> random_return(std::mt19937& gen, cell *from, cell *to, ld peq, ld pbad) {
 | |
|   vector<cell*> res = { from };
 | |
|   ld d = celldistance(to, from);
 | |
|   while(to != res.back()) {
 | |
|     int i = gen() % res.back()->type;
 | |
|     cell *r1 = res.back()->cmove(i);
 | |
|     ld d1 = celldistance(to, r1);
 | |
|     bool ok = d1 < d ? true : d1 == d ? rand01<ld>(gen) < peq : rand01<ld>(gen) < pbad;
 | |
|     if(ok) { res.push_back(r1); d = d1; }
 | |
|     }
 | |
|   return res;
 | |
|   }
 | |
| 
 | |
| vector<cell*> vrev(vector<cell*> a) { reverse(a.begin(), a.end()); return a; }
 | |
| vector<cell*> vcon(vector<cell*> a, vector<cell*> b) { for(auto bi: b) a.push_back(bi); return a; }
 | |
| 
 | |
| template<class N> N edge_of_triangle_with_angles(N alpha, N beta, N gamma) {
 | |
|   N of = (cos(alpha) + cos(beta) * cos(gamma)) / (sin(beta) * sin(gamma));
 | |
|   if(hyperbolic) return acosh(of);
 | |
|   return acos(of);
 | |
|   }
 | |
| 
 | |
| template<class N> N get_edgelen() {
 | |
|   N beta = get_deg<N>(360)/valence;
 | |
|   return edge_of_triangle_with_angles<N> (beta, get_deg<N>(180)/edges, get_deg<N>(180)/edges);
 | |
|   }
 | |
| 
 | |
| template<class T> typename T::isometry cpush(int c, typename T::data::Number distance) {
 | |
|   return T::lorentz(c, T::data::Dim-1, distance);
 | |
|   }
 | |
| 
 | |
| template<class T> struct cube_rotation_data_t {
 | |
|   std::vector<std::pair<hr::transmatrix, typename T::isometry>> mapping;
 | |
|   };
 | |
| 
 | |
| template<class T> cube_rotation_data_t<T> cube_rotation_data;
 | |
| 
 | |
| template<class T> cube_rotation_data_t<T>& build_cube_rotation() {
 | |
|   auto& crd = cube_rotation_data<T>;
 | |
|   auto& crdm = crd.mapping;
 | |
|   // using N = typename T::data::Number;
 | |
|   if(crdm.empty()) {
 | |
|     crdm.emplace_back(hr::Id, T::id());
 | |
|     for(int i=0; i<isize(crdm); i++) 
 | |
|     for(int j=0; j<3; j++) for(int k=0; k<3; k++) if(j != k) {
 | |
|       hr::transmatrix U = crdm[i].first * hr::cspin90(j, k);
 | |
|       bool is_new = true;
 | |
|       for(int i0=0; i0<isize(crdm); i0++) if(hr::eqmatrix(U, crdm[i0].first)) is_new = false;
 | |
|       // if(is_new) crdm.emplace_back(U, T::apply(crdm[i].second, T::cspin(j, k, get_deg<N>(90))));
 | |
|       if(is_new) crdm.emplace_back(U, T::apply(crdm[i].second, T::cspin90(j, k)));
 | |
|       }
 | |
|     if(isize(crdm) != 24) {
 | |
|       println(hlog, "the number of rotations found: ", isize(crdm));
 | |
|       throw hr::hr_exception("wrong number of rotations");
 | |
|       }
 | |
|     }
 | |
|   return crd;
 | |
|   }
 | |
| 
 | |
| template<class T, class U> U apply_move(cell *a, cell *b, U to) {
 | |
|   if(a == b) return to;
 | |
|   using N = typename T::data::Number;
 | |
|   if constexpr(test_dim == 4) {
 | |
|     auto& crdm = build_cube_rotation<T>().mapping;
 | |
|     int ida = neighborId(a, b);
 | |
|     auto M = hr::currentmap->adj(a, ida);
 | |
|     for(int i0=0; i0<isize(crdm); i0++)
 | |
|     for(int i1=0; i1<isize(crdm); i1++)
 | |
|       if(hr::eqmatrix(M, crdm[i0].first * hr::xpush(1.06128) * crdm[i1].first)) {
 | |
|         to = T::apply(crdm[i1].second, to);
 | |
|         to = T::apply(cpush<T>(0, get_edgelen<N>()), to);
 | |
|         to = T::apply(crdm[i0].second, to);
 | |
|         return to;
 | |
|         }
 | |
|     println(hlog, "tessf = ", hr::cgip->tessf);
 | |
|     println(hlog, "len = ", get_edgelen<N>());
 | |
|     throw hr::hr_exception("rotation not found");
 | |
|     }
 | |
|   int ida = neighborId(a, b);
 | |
|   int idb = neighborId(b, a);
 | |
|   auto P1 = T::cspin(0, 1, idb * get_deg<N>(360) / edges);
 | |
|   to = T::apply(P1, to);
 | |
|   auto P2 = cpush<T>(0, get_edgelen<N>());
 | |
|   to = T::apply(P2, to);
 | |
|   auto P3 = T::cspin(1, 0, get_deg<N>(180) + ida * get_deg<N>(360) / edges);
 | |
|   to = T::apply(P3, to);
 | |
|   return to;
 | |
|   }
 | |
| 
 | |
| template<class T, class U> U apply_path(vector<cell*> path, U to) {
 | |
|   for(int i=hr::isize(path)-2; i>=0; i--)
 | |
|     to = apply_move<T, U> (path[i], path[i+1], to);
 | |
|   return to;
 | |
|   }
 | |
| 
 | |
| template<class T> double test_sanity(int i) {
 | |
|   hr::indenter in(2);
 | |
| 
 | |
|   ld d1 = 1.25, d2 = 1.5;
 | |
| 
 | |
|   typename good::point gp = good::center();
 | |
|   gp = good::apply(cpush<good>(0, d1), gp);
 | |
|   gp = good::apply(cpush<good>(1, d2), gp);
 | |
|   gp = good::apply(good::cspin(0, 1, get_deg<big>(72)), gp);
 | |
| 
 | |
|   typename T::point p = T::center();
 | |
|   p = T::apply(cpush<T>(0, d1), p);
 | |
|   p = T::apply(cpush<T>(1, d2), p);
 | |
|   p = T::apply(T::cspin(0, 1, get_deg<typename T::data::Number>(72)), p);
 | |
| 
 | |
|   double res = 0;
 | |
|   #define ADD(x, y) if(debug) println(hlog, "VS ", x, ",", y); res += pow( double(x-y), 2);
 | |
|   #define ADDA(x, y) if(debug) println(hlog, "VS ", x, ",", y); res += pow( cyclefix_on(double(x-y)), 2);
 | |
| 
 | |
|   if(debug) println(hlog, "p=", T::print(p));
 | |
| 
 | |
|   ADD(T::get_coord(p, 0), good::get_coord(gp, 0));
 | |
|   ADD(T::get_coord(p, 1), good::get_coord(gp, 1));
 | |
|   ADD(T::get_coord(p, 2), good::get_coord(gp, 2));
 | |
|   ADD(T::dist0(p), good::dist0(gp));
 | |
|   ADDA(T::angle(p), good::angle(gp));
 | |
| 
 | |
|   return res;
 | |
|   }
 | |
| 
 | |
| template<class T> double test_consistency(int i) {
 | |
|   double res = 0;
 | |
|   using D = typename T::data;
 | |
| 
 | |
|   auto a = cpush<T>(0, 1);
 | |
|   auto b = cpush<T>(1, 1);
 | |
|   auto c = cpush<T>(0, 1);
 | |
| 
 | |
|   auto s   = T::apply(T::apply(a, b), c);
 | |
|   auto sp  = T::apply(s, T::center());
 | |
|   auto s1  = T::apply(a, T::apply(b, c));
 | |
|   auto sp1 = T::apply(s1, T::center());
 | |
|   auto sp2 = T::apply(a, T::apply(b, T::apply(c, T::center())));
 | |
|   ADD(T::dist0(sp), T::dist0(sp1));
 | |
|   ADD(T::dist0(sp), T::dist0(sp2));
 | |
|   for(int i=0; i<D::Dim; i++) { ADD(T::get_coord(sp, i), T::get_coord(sp1, i)); }
 | |
|   for(int i=0; i<D::Dim; i++) { ADD(T::get_coord(sp, i), T::get_coord(sp2, i)); }
 | |
|   if(test_dim == 3) {
 | |
|     ADDA(T::angle(sp), T::angle(sp1));
 | |
|     ADDA(T::angle(sp), T::angle(sp2));
 | |
|     }
 | |
| 
 | |
|   return res;
 | |
|   }
 | |
| 
 | |
| template<class T> double test_tba(int id) {
 | |
|   std::mt19937 testr; testr.seed(id);
 | |
|   for(int i=0; i<hr::cwt.at->type; i++) {
 | |
|     vector<cell*> p = {hr::cwt.at, hr::cwt.at->cmove(i), hr::cwt.at};
 | |
|     auto h = apply_path<T>(p, T::center());
 | |
|     if(debug == 2) println(hlog, "i=", hr::lalign(3, i), " h = ", T::print(h), " distance =", T::dist0(h));
 | |
|     if(T::dist0(h) >= 0 && T::dist0(h) < 0.1) continue;
 | |
|     exit(1);
 | |
|     }
 | |
|   return 999;
 | |
|   }
 | |
| 
 | |
| template<class T> double test_loop_point(int id) {
 | |
|   std::mt19937 testr; testr.seed(id);
 | |
|   for(int i=0; i<100; i++) {
 | |
|     auto p1 = randomwalk(testr, hr::cwt.at, i);
 | |
|     auto p2 = random_return(testr, p1.back(), hr::cwt.at, 1/16., 1/32.);
 | |
|     auto p = vcon(p1, p2);
 | |
|     if(debug == 2) println(hlog, "path = ", p);
 | |
|     auto h = apply_path<T>(vrev(p), T::center());
 | |
|     if(debug == 2) println(hlog, "i=", hr::lalign(3, i), " h = ", T::print(h), " distance =", T::dist0(h));
 | |
|     if(T::dist0(h) >= 0 && T::dist0(h) < 0.1) continue;
 | |
|     return i;
 | |
|     }
 | |
|   return 999;
 | |
|   }
 | |
| 
 | |
| template<class T> double test_loop_iso(int id) {
 | |
|   std::mt19937 testr; testr.seed(id);
 | |
|   for(int i=0; i<100; i++) {
 | |
|     auto p1 = randomwalk(testr, hr::cwt.at, i);
 | |
|     auto p2 = random_return(testr, p1.back(), hr::cwt.at, 1/16., 1/32.);
 | |
|     auto p = vcon(p1, p2);
 | |
|     if(debug == 2) println(hlog, "path = ", p);
 | |
|     auto h = apply_path<T>(vrev(p), T::id());
 | |
|     auto hr = T::apply(h, T::center());
 | |
|     // println(hlog, "i=", hr::lalign(3, i), " h=", hr);
 | |
|     if(debug == 2) println(hlog, "i=", hr::lalign(3, i), " hr = ", T::print(hr), " distance = ", T::dist0(hr));
 | |
|     if(T::dist0(hr) >= 0 && T::dist0(hr) < 0.1) continue;
 | |
|     if(debug == 1) println(hlog, "i=", hr::lalign(3, i), " hr = ", T::print(hr), " distance = ", T::dist0(hr));
 | |
|     return i;
 | |
|     }
 | |
|   return 999;
 | |
|   }
 | |
| 
 | |
| template<class T, class F> vector<T> repeat_test(const F& f, int qty) {
 | |
|   vector<T> res;
 | |
|   for(int i=0; i<qty; i++) res.push_back(f(i));
 | |
|   return res;
 | |
|   }
 | |
| 
 | |
| template<class T> typename T::isometry random_rotation(std::mt19937& testr) {
 | |
|   using D = typename T::data;
 | |
|   using N = typename D::Number;
 | |
| 
 | |
|   if(D::Dim == 3) {
 | |
|     auto alpha = rand01<N>(testr) * get_deg<N>(360);
 | |
|     return T::cspin(0, 1, alpha);
 | |
|     }
 | |
| 
 | |
|   auto x = T::id();
 | |
|   for(int i=0; i<100; i++) {
 | |
|     int c0 = testr() % (D::Dim-1);
 | |
|     int c1 = testr() % (D::Dim-1);
 | |
|     if(c0 == c1) continue;
 | |
|     auto len = rand01<N>(testr) * get_deg<N>(360);
 | |
|     x = T::apply(T::cspin(c0, c1, len), x);
 | |
|     }
 | |
|   return x;
 | |
|   }
 | |
| 
 | |
| template<class T> typename T::isometry random_iso(std::mt19937& testr) {
 | |
|   auto x = T::id();
 | |
|   using D = typename T::data;
 | |
|   using N = typename D::Number;
 | |
|   for(int i=0; i<100; i++) {
 | |
|     int c0 = testr() % D::Dim;
 | |
|     int c1 = testr() % D::Dim;
 | |
|     if(c0 == c1) continue;
 | |
|     if(c0 == D::Flipped) std::swap(c0, c1);
 | |
|     N len = c1 < D::Flipped ? rand01<N>(testr) * get_deg<N>(360) : (rand01<N>(testr)-N(0.5)) * N(0.25);
 | |
|     if(c1 == D::Flipped) x = T::apply(T::lorentz(c0, c1, len), x);
 | |
|     else x = T::apply(T::cspin(c0, c1, len), x);
 | |
|     }
 | |
|   return x;
 | |
|   }
 | |
| 
 | |
| 
 | |
| template<class T> std::string test_count(int id) {
 | |
|   std::mt19937 testr; testr.seed(id);
 | |
|   hr::shstream out;
 | |
|   auto A = random_iso<T>(testr);
 | |
|   auto B = random_iso<T>(testr);
 | |
|   auto C = random_iso<T>(testr);
 | |
|   auto P = T::apply(C, T::center());
 | |
|   for(int i=0; i<9; i++) {
 | |
|     counts.clear();
 | |
|     for(auto& i: cbc) i = 0;
 | |
|     std::string s;
 | |
|     switch(i) {
 | |
|       case 0: s = "spin"; T::cspin(0, 1, countfloat(.5)); break;
 | |
|       case 1: s = "L0"; T::lorentz(0, T::data::Dim-1, countfloat(.5)); break;
 | |
|       case 2: s = "L1"; T::lorentz(1, T::data::Dim-1, countfloat(.5)); break;
 | |
|       case 3: s = "ip"; T::apply(A, P); break;
 | |
|       case 4: s = "ii"; T::apply(A, B); break;
 | |
|       case 5: s = "d0"; T::dist0(P); break;
 | |
|       case 6: s = "angle"; T::angle(P); break;
 | |
|       case 7: s = "inverse"; T::inverse(A); break;
 | |
|       case 8: s = "push"; T::push(P); break;
 | |
|       }
 | |
|     if(i) print(out, " ");
 | |
|     if(1) {
 | |
|       print(out, s, "(");
 | |
|       bool nsp = false;
 | |
|       for(int i=1; i<5; i++) if(cbc[i]) {
 | |
|         if(nsp) print(out, " ");
 | |
|         print(out, cbc[i], hr::s0+".AMDF"[i]);
 | |
|         nsp = true;
 | |
|         }
 | |
|       print(out, ")");
 | |
|       }
 | |
|     }
 | |
|   return out.s;
 | |
|   }
 | |
| 
 | |
| template<class A, class B> bool closeto(A a, B b) { return abs(a-b) < 0.1; }
 | |
| 
 | |
| template<class A, class B> bool closeto_angle(A a, B b) { return abs(cyclefix_on(double(a-b))) < 0.1; }
 | |
| 
 | |
| template<class T> double test_angledist(int id) {
 | |
|   std::mt19937 testr; testr.seed(id);
 | |
|   for(int i=1; i<1000; i += (1 + i/5)) {
 | |
| 
 | |
|     auto p = randomwalk(testr, hr::cwt.at, i);
 | |
|     auto h = apply_path<T>(vrev(p), T::center());
 | |
| 
 | |
|     auto gh = apply_path<good>(vrev(p), good::center());
 | |
| 
 | |
|     if(debug == 2) {
 | |
|       println(hlog, "good: ", good::print(gh), " dist = ", good::dist0(gh), " angle = ", good::angle(gh));
 | |
|       println(hlog, "test: ", T::print(h), " dist = ", T::dist0(h), " angle = ", T::angle(h), " [i=", i, "]");
 | |
|       }
 | |
| 
 | |
|     if(closeto(good::dist0(gh), T::dist0(h)) && closeto_angle(good::angle(gh), T::angle(h))) continue;
 | |
| 
 | |
|     if(debug == 1) {
 | |
|       println(hlog, "good: ", good::print(gh), " dist = ", good::dist0(gh), " angle = ", good::angle(gh));
 | |
|       println(hlog, "test: ", T::print(h), " dist = ", T::dist0(h), " angle = ", T::angle(h), " [i=", i, "]");
 | |
|       }
 | |
| 
 | |
|     return i;
 | |
|     }
 | |
|   return 999;
 | |
|   }
 | |
| 
 | |
| #define TEST_VARIANTS(x,D,q,t,rn, r) \
 | |
|   nm = nmInvariant; println(hlog, rn, "invariant: ", repeat_test<t>(x<r<D>>, q));  \
 | |
|   nm = nmForced;    println(hlog, rn, "forced   : ", repeat_test<t>(x<r<D>>, q));   \
 | |
|   nm = nmWeak;      println(hlog, rn, "weak     : ", repeat_test<t>(x<r<D>>, q));    \
 | |
|   nm = nmFlatten;   println(hlog, rn, "flatten  : ", repeat_test<t>(x<r<D>>, q));     \
 | |
|   nm = nmCareless;  println(hlog, rn, "careless : ", repeat_test<t>(x<r<D>>, q));      \
 | |
|   nm = nmBinary;    println(hlog, rn, "binary   : ", repeat_test<t>(x<r<D>>, q));
 | |
| 
 | |
| /*
 | |
| #define TEST_ALL(x,D,q,t) \
 | |
|   println(hlog, "HyperRogue: ", repeat_test<t>(x<rep_hr<D>>, q));                                             \
 | |
|   polar_mod = polar_choose = false;  println(hlog, "high polar: ", repeat_test<t>(x<rep_high_polar<D>>, q));   \
 | |
|   if(test_dim == 3) { polar_mod = polar_choose = false;  println(hlog, "low polar : ", repeat_test<t>(x<rep_polar2<D>>, q)); }
 | |
| */
 | |
| 
 | |
| //                    println(hlog, "HyperRogue         : ", repeat_test<t>(x<rep_hr<D>>, q));                     
 | |
| 
 | |
| #define TEST_ALL(x,D,q,t) \
 | |
|   fix_matrices = true;  TEST_VARIANTS(x,D,q,t,"linear+F  ", rep_linear)                                           \
 | |
|   fix_matrices = false; TEST_VARIANTS(x,D,q,t,"linear-F  ", rep_linear)                                            \
 | |
|   TEST_VARIANTS(x,D,q,t,"mixed     ", rep_mixed)                                                                    \
 | |
|   TEST_VARIANTS(x,D,q,t,"Clifford  ", rep_clifford)                                                                  \
 | |
|   nm = nmFlatten; println(hlog, "Clifford  gyro     : ", repeat_test<t>(x<rep_half<D>>, q));                          \
 | |
|   nm = nmInvariant; println(hlog, "halfplane invariant: ", repeat_test<t>(x<rep_half<D>>, q));                         \
 | |
|   polar_choose = false;  println(hlog, "polar     basic    : ", repeat_test<t>(x<rep_high_polar<D>>, q));               \
 | |
|   polar_choose = true;   println(hlog, "polar     improved : ", repeat_test<t>(x<rep_high_polar<D>>, q));                \
 | |
|   if(test_dim == 3) {                                                                                                     \
 | |
|     polar_mod = false; polar_choose = false;  println(hlog, "polar     F/F      : ", repeat_test<t>(x<rep_polar2<D>>, q)); \
 | |
|     polar_mod = false; polar_choose = true;   println(hlog, "polar     F/T      : ", repeat_test<t>(x<rep_polar2<D>>, q));  \
 | |
|     polar_mod = true;  polar_choose = false;  println(hlog, "polar     T/F      : ", repeat_test<t>(x<rep_polar2<D>>, q));   \
 | |
|     polar_mod = true;  polar_choose = true;   println(hlog, "polar     T/T      : ", repeat_test<t>(x<rep_polar2<D>>, q));    \
 | |
|     }
 | |
| 
 | |
| template<class T> double test_distances(int id, int a) {
 | |
|   std::mt19937 testr; testr.seed(id);
 | |
|   using N = typename T::data::Number;
 | |
| 
 | |
|   for(int i=1; i<1000; i ++) {
 | |
| 
 | |
|     auto R = random_rotation<T>(testr);
 | |
|     auto dif = exp(N(-1) * i) + get_deg<N>(a);
 | |
| 
 | |
|     auto p1 = T::apply(T::apply(R, cpush<T>(0, N(i))), T::center());
 | |
|     auto p2 = T::apply(T::apply(R, T::apply(T::cspin(0, 1, dif), cpush<T>(0, N(i)))), T::center());
 | |
|     auto pd = T::apply(T::inverse(T::push(p1)), p2);
 | |
|     auto d = T::dist0(pd);
 | |
| 
 | |
|     // for good we do not need R actually
 | |
|     auto gp1 = good::apply(cpush<good>(0, N(i)), good::center());
 | |
|     auto gp2 = good::apply(good::apply(good::cspin(0, 1, dif), cpush<good>(0, N(i))), good::center());
 | |
|     auto gd = good::dist0(good::apply(good::inverse(good::push(gp1)), gp2));
 | |
| 
 | |
|     if(debug == 2) println(hlog, T::print(p1), " ... ", T::print(p2), " = ", T::print(pd), " d=", d, " [i=", i, " dif=", dif, "]");
 | |
| 
 | |
|     if(closeto(d, gd)) continue;
 | |
| 
 | |
|     return i;
 | |
|     }
 | |
|   return 999;
 | |
|   }
 | |
| 
 | |
| template<class T> double test_similarity(int id) { return test_distances<T>(id, 0); }
 | |
| template<class T> double test_dissimilarity(int id) { return test_distances<T>(id, 180); }
 | |
| template<class T> double test_other(int id) { return test_distances<T>(id, 1); }
 | |
| 
 | |
| template<class T> double test_walk(int id) {
 | |
|   std::mt19937 testr; testr.seed(id);
 | |
| 
 | |
|   ld step = 1/16.;
 | |
|   // mover-relative to cell-relative
 | |
|   auto R0 = random_rotation<T>(testr);
 | |
|   cell *c0 = hr::cwt.at;
 | |
|   auto R1 = T::apply(R0, cpush<T>(0, step/2));
 | |
|   cell *c1 = hr::cwt.at;
 | |
| 
 | |
|   int i = 0;
 | |
|   int lastchange = 0;
 | |
|   while(i< lastchange + 1000 && i < 10000 && celldistance(c0, c1) < 3) {
 | |
|     // println(hlog, "iteration ", i, " in ", c0, " vs ", c1);
 | |
|     auto rebase = [&] (typename T::isometry& R, cell*& c, int id) {
 | |
|       ld d = T::dist0(T::apply(R, T::center()));
 | |
|       for(int dir=0; dir<c->type; dir++) {
 | |
|         cell *altc = c->cmove(dir);
 | |
|         auto altR = apply_move<T>(altc, c, R);
 | |
|         ld altd = T::dist0(T::apply(altR, T::center()));
 | |
|         if(altd < d + 1/256.) {
 | |
|           R = altR; c = altc; lastchange = i; return;
 | |
|           }
 | |
|         }
 | |
|       };
 | |
|     R0 = T::apply(R0, cpush<T>(0, step)); rebase(R0, c0, 0);
 | |
|     R1 = T::apply(R1, cpush<T>(0, step)); rebase(R1, c1, 1);
 | |
|     i++;
 | |
|     }
 | |
| 
 | |
|   return i;
 | |
|   }
 | |
| 
 | |
| template<class T> double test_close(int id) {
 | |
|   std::mt19937 testr; testr.seed(id);
 | |
| 
 | |
|   cell *c = hr::cwt.at;
 | |
|   int phase = 0;
 | |
|   auto p0 = T::apply(cpush<T>(0, 1/8.), T::center());
 | |
|   auto p = p0;
 | |
| 
 | |
|   int steps = 0;
 | |
|   const int maxdist = id + 1;
 | |
|   int errors = 0; 
 | |
| 
 | |
|   while(steps < 10000) {
 | |
|     int d = testr() % c->type;
 | |
|     cell *c1 = c->cmove(d);
 | |
| 
 | |
|     bool do_move = false;
 | |
| 
 | |
|     switch(phase) {
 | |
|       case 0:
 | |
|         /* always move */
 | |
|         do_move = true;
 | |
|         if(celldistance(c1, hr::cwt.at) == maxdist) phase = 1;
 | |
|         break;
 | |
| 
 | |
|       case 1:
 | |
|         /* move only towards the center */
 | |
|         int d0 = celldistance(c, hr::cwt.at);
 | |
|         int d1 = celldistance(c1, hr::cwt.at);
 | |
|         do_move = d1 < d0;
 | |
|         if(d1 == 0) phase = 0;
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|     if(do_move) {
 | |
|       p = apply_move<T>(c1, c, p); c = c1; steps++;
 | |
|       if(debug == 2) println(hlog, "dist = ", celldistance(c, hr::cwt.at), " dist = ", T::dist0(p));
 | |
|       if(c == hr::cwt.at) {
 | |
|         auto d = T::dist0(p);
 | |
|         auto a = T::angle(p);
 | |
|         if(!closeto(d, 1/8.) || !closeto_angle(a, 0)) {
 | |
|           errors++; phase = 0; c = hr::cwt.at; p = p0;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|   return errors;
 | |
|   }
 | |
| 
 | |
| void run_all_tests() {
 | |
|   prepare_tests();
 | |
| 
 | |
|   // println(hlog, "test_sanity"); TEST_ALL(test_sanity, data, 1, ld);
 | |
| 
 | |
|   // println(hlog, "test_consistency"); TEST_ALL(test_consistency, data, 1, ld);
 | |
| 
 | |
|   println(hlog, "test_loop_iso"); TEST_ALL(test_loop_iso, data, 20, int);
 | |
| 
 | |
|   println(hlog, "test_loop_point"); TEST_ALL(test_loop_point, data, 20, int);
 | |
| 
 | |
|   println(hlog, "test_angledist"); TEST_ALL(test_angledist, data, 3, int);
 | |
| 
 | |
|   println(hlog, "test_similarity"); TEST_ALL(test_similarity, data, 20, int);
 | |
| 
 | |
|   println(hlog, "test_dissimilarity"); TEST_ALL(test_dissimilarity, data, 20, int);
 | |
| 
 | |
|   println(hlog, "test_other"); TEST_ALL(test_other, data, 20, int);
 | |
| 
 | |
|   println(hlog, "test_walk"); TEST_ALL(test_walk, data, 20, int);
 | |
| 
 | |
|   println(hlog, "test_close"); TEST_ALL(test_close, data, 20, int);
 | |
| 
 | |
|   println(hlog, "test_count"); TEST_ALL(test_count, countdata, 1, std::string);
 | |
|   }
 | |
| 
 | |
| }
 | 
