line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
2
|
|
|
2
|
|
1154
|
use strict; use warnings; |
|
2
|
|
|
2
|
|
3
|
|
|
2
|
|
|
|
|
57
|
|
|
2
|
|
|
|
|
6
|
|
|
2
|
|
|
|
|
3
|
|
|
2
|
|
|
|
|
1935
|
|
2
|
|
|
|
|
|
|
package Language::SNUSP; |
3
|
|
|
|
|
|
|
our $VERSION = '0.0.14'; |
4
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
my $input = ''; # SNUSP input |
6
|
|
|
|
|
|
|
my $code = ''; # 2D code matrix |
7
|
|
|
|
|
|
|
my $width = 1; # 2D code width |
8
|
|
|
|
|
|
|
my $pos = 0; # 2D code execution pointer |
9
|
|
|
|
|
|
|
my $max = 0; # Maximum pos value (length of code) |
10
|
|
|
|
|
|
|
my $dir = 1; # Execution direction: |
11
|
|
|
|
|
|
|
# 1=right -1=left $width=down -$width=up |
12
|
|
|
|
|
|
|
my @args = (); # Program input list |
13
|
|
|
|
|
|
|
my @data = (0); # Data slots |
14
|
|
|
|
|
|
|
my $index = 0; # Data slot index |
15
|
|
|
|
|
|
|
my @stack = (); # Subroutine call stack |
16
|
|
|
|
|
|
|
my $count = 0; # Execution counter |
17
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
# I/O handlers |
19
|
|
|
|
|
|
|
my $put = sub { print shift }; |
20
|
|
|
|
|
|
|
my $get = sub { substr shift(@args), 0, 1 }; |
21
|
|
|
|
|
|
|
|
22
|
|
|
|
|
|
|
# SNUSP opcode handler lookup table. |
23
|
|
|
|
|
|
|
my %ops = ( |
24
|
|
|
|
|
|
|
'>' => sub { $data[++$index] ||= 0 }, |
25
|
|
|
|
|
|
|
'<' => sub { --$index >= 0 or $dir = 0 }, |
26
|
|
|
|
|
|
|
'+' => sub { ++$data[$index] }, |
27
|
|
|
|
|
|
|
'-' => sub { --$data[$index] }, |
28
|
|
|
|
|
|
|
',' => sub { $data[$index] = ord $get->() }, |
29
|
|
|
|
|
|
|
'.' => sub { $put->(chr $data[$index]) }, |
30
|
|
|
|
|
|
|
'/' => sub { $dir = -$width / $dir }, |
31
|
|
|
|
|
|
|
'\\' => sub { $dir = $width / $dir }, |
32
|
|
|
|
|
|
|
'!' => sub { $pos += $dir }, |
33
|
|
|
|
|
|
|
'?' => sub { $pos += $dir if $data[$index] == 0 }, |
34
|
|
|
|
|
|
|
'@' => sub { push @stack, [ $pos + $dir, $dir ] }, |
35
|
|
|
|
|
|
|
'#' => sub { @stack ? ($pos, $dir) = @{pop @stack} : $dir = 0 }, |
36
|
|
|
|
|
|
|
"\n" => sub { $dir = 0 }, |
37
|
|
|
|
|
|
|
); |
38
|
|
|
|
|
|
|
|
39
|
|
|
|
|
|
|
# Runtime flags |
40
|
|
|
|
|
|
|
my $file; # Input SNUSP file |
41
|
|
|
|
|
|
|
my $trace = 0; # Run with trace execution |
42
|
|
|
|
|
|
|
my $debug = 0; # Run with 2D Curses debugger |
43
|
|
|
|
|
|
|
|
44
|
|
|
|
|
|
|
sub run { |
45
|
0
|
|
|
0
|
0
|
|
my ($class, @args) = @_; |
46
|
0
|
|
|
|
|
|
$class->get_options(@args); |
47
|
|
|
|
|
|
|
|
48
|
0
|
0
|
|
|
|
|
open my $fh, '<', $file or die "Can't open '$file' for input.\n"; |
49
|
0
|
|
|
|
|
|
$input = do { local $/; <$fh> }; |
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
50
|
0
|
|
|
|
|
|
close $fh; |
51
|
|
|
|
|
|
|
|
52
|
0
|
|
|
|
|
|
for ($input =~ /^.*\n/gm) { |
53
|
0
|
|
|
|
|
|
$code .= $_; |
54
|
0
|
0
|
|
|
|
|
$width = length if length > $width; |
55
|
|
|
|
|
|
|
} |
56
|
0
|
|
|
|
|
|
$code =~ s/^.*/$& . ' ' x ($width - length $&) . "\n"/gem; |
|
0
|
|
|
|
|
|
|
57
|
0
|
|
|
|
|
|
$max = length($code) - 1; |
58
|
0
|
|
|
|
|
|
$width += 2; |
59
|
0
|
|
|
|
|
|
$pos = $code =~ /\$/ * $-[0]; |
60
|
|
|
|
|
|
|
|
61
|
0
|
0
|
|
|
|
|
$trace ? run_trace() : |
|
|
0
|
|
|
|
|
|
62
|
|
|
|
|
|
|
$debug ? run_debug() : |
63
|
|
|
|
|
|
|
run_normal(); |
64
|
|
|
|
|
|
|
|
65
|
0
|
|
|
|
|
|
exit $data[$index]; |
66
|
|
|
|
|
|
|
} |
67
|
|
|
|
|
|
|
|
68
|
|
|
|
|
|
|
sub run_normal { |
69
|
0
|
|
|
0
|
0
|
|
while ($dir) { |
70
|
0
|
0
|
|
|
|
|
if (my $op = $ops{substr $code, $pos, 1}) { &$op } |
|
0
|
|
|
|
|
|
|
71
|
0
|
|
|
|
|
|
$pos += $dir; |
72
|
0
|
0
|
0
|
|
|
|
last if $pos < 0 or $pos > $max; |
73
|
|
|
|
|
|
|
} |
74
|
|
|
|
|
|
|
} |
75
|
|
|
|
|
|
|
|
76
|
|
|
|
|
|
|
sub run_trace { |
77
|
0
|
|
|
0
|
0
|
|
while ($dir) { |
78
|
0
|
|
|
|
|
|
my $char = substr $code, $pos, 1; |
79
|
0
|
|
|
|
|
|
$count++; |
80
|
0
|
|
|
|
|
|
print trace_line() . "\n"; |
81
|
0
|
0
|
|
|
|
|
if (my $op = $ops{$char}) { &$op } |
|
0
|
|
|
|
|
|
|
82
|
0
|
|
|
|
|
|
$pos += $dir; |
83
|
0
|
0
|
0
|
|
|
|
last if $pos < 0 or $pos > $max; |
84
|
0
|
0
|
|
|
|
|
print "\n" if $char eq '.'; |
85
|
|
|
|
|
|
|
} |
86
|
|
|
|
|
|
|
} |
87
|
|
|
|
|
|
|
|
88
|
|
|
|
|
|
|
sub run_debug { |
89
|
0
|
|
|
0
|
0
|
|
require Curses; Curses->import; |
|
0
|
|
|
|
|
|
|
90
|
0
|
|
|
|
|
|
require Term::ReadKey; Term::ReadKey->import; |
|
0
|
|
|
|
|
|
|
91
|
|
|
|
|
|
|
|
92
|
0
|
|
|
|
|
|
initscr(); |
93
|
0
|
|
|
|
|
|
ReadMode(3); |
94
|
|
|
|
|
|
|
|
95
|
0
|
|
|
|
|
|
my $y = 0; |
96
|
0
|
|
|
|
|
|
addstr( |
97
|
|
|
|
|
|
|
$y++, 0, |
98
|
|
|
|
|
|
|
"(n)ext (SPACE)stop/start (+)faster (-)slower (q)uit", |
99
|
|
|
|
|
|
|
); |
100
|
0
|
|
|
|
|
|
my $top = ++$y; |
101
|
0
|
|
|
|
|
|
addstr($y++, 0, $&) while $code =~ /.+/g; |
102
|
|
|
|
|
|
|
|
103
|
0
|
|
|
|
|
|
my $key = ''; |
104
|
0
|
|
|
|
|
|
my $sleep = 0.1; |
105
|
0
|
|
|
|
|
|
my $pause = 1; |
106
|
|
|
|
|
|
|
|
107
|
0
|
|
|
|
|
|
my $out = ''; |
108
|
0
|
|
|
0
|
|
|
$put = sub { $out .= shift }; |
|
0
|
|
|
|
|
|
|
109
|
|
|
|
|
|
|
|
110
|
0
|
|
|
|
|
|
while(1) { |
111
|
0
|
0
|
0
|
|
|
|
if ($dir and (not $pause or $key eq "n")) { |
|
|
|
0
|
|
|
|
|
112
|
0
|
|
|
|
|
|
$count++; |
113
|
0
|
0
|
|
|
|
|
if (my $op = $ops{substr $code, $pos, 1}) { &$op } |
|
0
|
|
|
|
|
|
|
114
|
0
|
0
|
0
|
|
|
|
last if $pos < 0 or $pos > $max; |
115
|
0
|
|
|
|
|
|
$pos += $dir; |
116
|
0
|
0
|
|
|
|
|
$pause = 1 if $dir == 0; |
117
|
|
|
|
|
|
|
} |
118
|
|
|
|
|
|
|
|
119
|
|
|
|
|
|
|
{ |
120
|
0
|
|
|
|
|
|
addstr($top - 1, 0, trace_line()); |
|
0
|
|
|
|
|
|
|
121
|
0
|
|
|
|
|
|
addstr($y, 0, $out); |
122
|
0
|
|
|
|
|
|
clrtoeol(); |
123
|
0
|
|
|
|
|
|
move(int($pos / $width) + $top, $pos % $width); |
124
|
0
|
|
|
|
|
|
refresh(); |
125
|
|
|
|
|
|
|
} |
126
|
|
|
|
|
|
|
|
127
|
2
|
|
|
2
|
|
16
|
no warnings 'uninitialized'; |
|
2
|
|
|
|
|
1
|
|
|
2
|
|
|
|
|
604
|
|
128
|
0
|
0
|
|
|
|
|
$key = ReadKey($pause ? 0 : $sleep); |
129
|
0
|
0
|
|
|
|
|
if ($key =~ /^[\+\=]$/) {$sleep -= 0.01 if $sleep > 0.011} |
|
0
|
0
|
|
|
|
|
|
|
0
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
130
|
0
|
|
|
|
|
|
elsif ($key eq '-') {$sleep += 0.01} |
131
|
0
|
|
|
|
|
|
elsif ($key eq ' ') {$pause = not $pause} |
132
|
0
|
|
|
|
|
|
elsif ($key eq 'n') {$pause = 1} |
133
|
|
|
|
|
|
|
elsif ($key eq 'q') {last} |
134
|
|
|
|
|
|
|
} |
135
|
0
|
|
|
|
|
|
ReadMode(0); |
136
|
0
|
|
|
|
|
|
endwin(); |
137
|
|
|
|
|
|
|
} |
138
|
|
|
|
|
|
|
|
139
|
|
|
|
|
|
|
sub trace_line { |
140
|
0
|
|
|
0
|
0
|
|
my $n = 0; |
141
|
0
|
0
|
|
|
|
|
my $display = join '', map { |
142
|
0
|
|
|
|
|
|
$n++ == $index ? "[$_] " : "$_ " |
143
|
|
|
|
|
|
|
} @data; |
144
|
0
|
|
|
|
|
|
return "$count) \@${\scalar @stack} < $display>"; |
|
0
|
|
|
|
|
|
|
145
|
|
|
|
|
|
|
} |
146
|
|
|
|
|
|
|
|
147
|
|
|
|
|
|
|
sub get_options { |
148
|
0
|
|
|
0
|
0
|
|
my ($class, @options) = @_; |
149
|
|
|
|
|
|
|
|
150
|
0
|
|
|
|
|
|
for my $option (@options) { |
151
|
0
|
0
|
|
|
|
|
if ($option =~ /^(-v|--version)$/) { |
152
|
2
|
|
|
2
|
|
8
|
no strict 'refs'; |
|
2
|
|
|
|
|
3
|
|
|
2
|
|
|
|
|
541
|
|
153
|
0
|
|
|
|
|
|
print qq!Language::SNUSP v${"VERSION"}!; |
|
0
|
|
|
|
|
|
|
154
|
0
|
|
|
|
|
|
exit 0; |
155
|
|
|
|
|
|
|
} |
156
|
0
|
0
|
|
|
|
|
if ($option =~ /^(-\?|-h|--help)$/) { |
157
|
0
|
|
|
|
|
|
die usage(); |
158
|
0
|
|
|
|
|
|
exit 0; |
159
|
|
|
|
|
|
|
} |
160
|
0
|
0
|
|
|
|
|
if ($option =~ /^(-d|--debug)$/) { |
161
|
0
|
|
|
|
|
|
$debug = 1; |
162
|
0
|
|
|
|
|
|
next; |
163
|
|
|
|
|
|
|
} |
164
|
0
|
0
|
|
|
|
|
if ($option =~ /^(-t|--trace)$/) { |
165
|
0
|
|
|
|
|
|
$trace = 1; |
166
|
0
|
|
|
|
|
|
next; |
167
|
|
|
|
|
|
|
} |
168
|
0
|
0
|
|
|
|
|
if ($option =~ /^-/) { |
169
|
0
|
|
|
|
|
|
die "Unknown option: '$option'\n\n" . usage(); |
170
|
|
|
|
|
|
|
} |
171
|
0
|
0
|
|
|
|
|
if ($file) { |
172
|
0
|
|
|
|
|
|
push @args, $option; |
173
|
0
|
|
|
|
|
|
next; |
174
|
|
|
|
|
|
|
} |
175
|
0
|
0
|
|
|
|
|
if (-f $option) { |
176
|
0
|
|
|
|
|
|
$file = $option; |
177
|
|
|
|
|
|
|
} |
178
|
|
|
|
|
|
|
else { |
179
|
0
|
|
|
|
|
|
die "Input file '$option' does not exist.\n"; |
180
|
|
|
|
|
|
|
} |
181
|
|
|
|
|
|
|
} |
182
|
0
|
0
|
|
|
|
|
die usage() if not $file; |
183
|
|
|
|
|
|
|
} |
184
|
|
|
|
|
|
|
|
185
|
|
|
|
|
|
|
sub usage { |
186
|
0
|
|
|
0
|
0
|
|
<<'...'; |
187
|
|
|
|
|
|
|
Usage: |
188
|
|
|
|
|
|
|
snusp [options] input_file.snusp |
189
|
|
|
|
|
|
|
|
190
|
|
|
|
|
|
|
Options: |
191
|
|
|
|
|
|
|
-d, --debug # Run program in the visual debugger |
192
|
|
|
|
|
|
|
-t, --trace # Run with trace on |
193
|
|
|
|
|
|
|
-v, --version # Print version and exit |
194
|
|
|
|
|
|
|
-h, --help # Print help and exit |
195
|
|
|
|
|
|
|
... |
196
|
|
|
|
|
|
|
} |
197
|
|
|
|
|
|
|
|
198
|
|
|
|
|
|
|
1; |