line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Path::Hilbert::BigInt; |
2
|
|
|
|
|
|
|
|
3
|
1
|
|
|
1
|
|
543
|
use strict; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
25
|
|
4
|
1
|
|
|
1
|
|
4
|
use warnings; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
22
|
|
5
|
1
|
|
|
1
|
|
17
|
use 5.012; |
|
1
|
|
|
|
|
3
|
|
6
|
1
|
|
|
1
|
|
4
|
use utf8; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
4
|
|
7
|
|
|
|
|
|
|
|
8
|
1
|
|
|
1
|
|
4539956
|
use Math::BigRat try => 'GMP,Pari,Calc'; |
|
1
|
|
|
|
|
349843
|
|
|
1
|
|
|
|
|
5
|
|
9
|
1
|
|
|
1
|
|
2163
|
use Math::BigInt try => 'GMP,Pari,Calc'; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
5
|
|
10
|
|
|
|
|
|
|
|
11
|
1
|
|
|
1
|
|
1115
|
use Exporter qw( import ); |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
816
|
|
12
|
|
|
|
|
|
|
|
13
|
|
|
|
|
|
|
our @EXPORT = qw( xy2d d2xy ); |
14
|
|
|
|
|
|
|
|
15
|
|
|
|
|
|
|
our $VERSION = '1.204'; |
16
|
|
|
|
|
|
|
|
17
|
|
|
|
|
|
|
# optional constructor if you want OO-style |
18
|
|
|
|
|
|
|
sub new { |
19
|
0
|
|
|
0
|
0
|
0
|
my $class = shift; |
20
|
0
|
|
|
|
|
0
|
my ($n) = @_; |
21
|
0
|
|
|
|
|
0
|
return bless { n => $n } => $class; |
22
|
|
|
|
|
|
|
} |
23
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
# convert (x,y) to d |
25
|
|
|
|
|
|
|
sub xy2d { |
26
|
1360
|
|
|
1360
|
0
|
185052
|
my ($side, $x, $y) = @_; |
27
|
1360
|
|
|
|
|
2819
|
my $n = _valid_n($side); |
28
|
1360
|
|
|
|
|
38066
|
($x, $y) = map { Math::BigInt->new("$_") } ($x, $y); |
|
2720
|
|
|
|
|
52225
|
|
29
|
1360
|
|
|
|
|
48392
|
my $d = Math::BigInt->bzero(); |
30
|
1360
|
|
|
|
|
27268
|
for (my $s = $n->copy()->brsft(1); $s->bcmp(0) > 0; $s->brsft(1)) { |
31
|
19720
|
100
|
|
|
|
4667081
|
my $rx = Math::BigInt->new($x->copy()->band($s)->bcmp(0) > 0 ? "1" : "0"); |
32
|
19720
|
100
|
|
|
|
5227850
|
my $ry = Math::BigInt->new($y->copy()->band($s)->bcmp(0) > 0 ? "1" : "0"); |
33
|
19720
|
|
|
|
|
5264120
|
my $three_rx = $rx->copy()->bmul("3"); |
34
|
19720
|
|
|
|
|
2269182
|
my $s_squared = $s->copy()->bpow("2"); |
35
|
19720
|
|
|
|
|
2863692
|
$d->badd($s_squared->bmul($three_rx->bxor($ry))); |
36
|
19720
|
|
|
|
|
2702314
|
($x, $y) = _rot($s, $x, $y, $rx, $ry); |
37
|
|
|
|
|
|
|
} |
38
|
1360
|
|
|
|
|
322368
|
return Math::BigRat->new($d); |
39
|
|
|
|
|
|
|
} |
40
|
|
|
|
|
|
|
|
41
|
|
|
|
|
|
|
# convert d to (x,y) |
42
|
|
|
|
|
|
|
sub d2xy { |
43
|
1360
|
|
|
1360
|
0
|
5730
|
my ($side, $d) = @_; |
44
|
1360
|
|
|
|
|
2957
|
my $n = _valid_n($side); |
45
|
1360
|
|
|
|
|
45635
|
my $t = Math::BigInt->new($d); |
46
|
1360
|
|
|
|
|
38109
|
my ($x, $y) = map { Math::BigInt->bzero() } (1 .. 2); |
|
2720
|
|
|
|
|
35230
|
|
47
|
1360
|
|
|
|
|
28999
|
for (my $s = Math::BigInt->bone(); $s->bcmp($n) < 0; $s->blsft(1)) { |
48
|
19720
|
|
|
|
|
4433725
|
my $rx = $t->copy()->brsft(1)->band(Math::BigInt->bone()); |
49
|
19720
|
|
|
|
|
3727277
|
my $ry = $t->copy()->bxor($rx)->band(Math::BigInt->bone()); |
50
|
19720
|
|
|
|
|
2784896
|
($x, $y) = _rot($s, $x, $y, $rx, $ry); |
51
|
19720
|
|
|
|
|
57161
|
my $Dx = $s->copy()->bmul($rx); |
52
|
19720
|
|
|
|
|
1216297
|
my $Dy = $s->copy()->bmul($ry); |
53
|
19720
|
50
|
|
|
|
1182227
|
$Dx >= 0 ? $x->badd($Dx) : $x->bsub($Dx->copy()->babs()); |
54
|
19720
|
50
|
|
|
|
3115115
|
$Dy >= 0 ? $y->badd($Dy) : $y->bsub($Dy->copy()->babs()); |
55
|
19720
|
|
|
|
|
3091742
|
$t->brsft(2); |
56
|
|
|
|
|
|
|
} |
57
|
1360
|
|
|
|
|
310827
|
return map { Math::BigRat->new($_) } ($x, $y); |
|
2720
|
|
|
|
|
148308
|
|
58
|
|
|
|
|
|
|
} |
59
|
|
|
|
|
|
|
|
60
|
|
|
|
|
|
|
# rotate/flip a quadrant appropriately |
61
|
|
|
|
|
|
|
sub _rot { |
62
|
39440
|
|
|
39440
|
|
68100
|
my ($n, $x, $y, $rx, $ry) = @_; |
63
|
39440
|
100
|
|
|
|
102881
|
if (!$ry) { |
64
|
36134
|
100
|
|
|
|
966113
|
if ($rx > 0) { |
65
|
1224
|
|
|
|
|
146986
|
$x = $n - 1 - $x; |
66
|
1224
|
|
|
|
|
277423
|
$y = $n - 1 - $y; |
67
|
|
|
|
|
|
|
} |
68
|
36134
|
|
|
|
|
4339900
|
($x, $y) = ($y, $x); |
69
|
|
|
|
|
|
|
} |
70
|
39440
|
|
|
|
|
250920
|
return ($x, $y); |
71
|
|
|
|
|
|
|
} |
72
|
|
|
|
|
|
|
|
73
|
|
|
|
|
|
|
sub _valid_n { |
74
|
2720
|
|
|
2720
|
|
5144
|
my $n = _extract_side(shift(@_)); |
75
|
2720
|
|
100
|
|
|
4555
|
$n = 2 ** int((eval { (log($n) / log(2)) } || 0) + 0.5); |
76
|
2720
|
|
|
|
|
9494
|
return Math::BigInt->new(int($n)); |
77
|
|
|
|
|
|
|
} |
78
|
|
|
|
|
|
|
|
79
|
|
|
|
|
|
|
sub _extract_side { |
80
|
2720
|
|
|
2720
|
|
3582
|
my ($side) = @_; |
81
|
2720
|
0
|
33
|
|
|
5734
|
$side = $side->{ n } if ref($side) eq 'HASH' && exists $side->{ n }; |
82
|
2720
|
|
|
|
|
4380
|
return $side; |
83
|
|
|
|
|
|
|
} |
84
|
|
|
|
|
|
|
|
85
|
|
|
|
|
|
|
1; |
86
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
__END__ |