forked from kframework/c-semantics
-
Notifications
You must be signed in to change notification settings - Fork 0
/
program-runner
executable file
·274 lines (221 loc) · 7.91 KB
/
program-runner
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
#!/usr/bin/env perl
use strict;
use warnings;
use List::Util qw(reduce);
use File::Spec::Functions qw(catfile);
use File::Temp;
use File::Copy;
use MIME::Base64;
use String::Escape qw(quote backslash);
setpgrp;
# We trap control-c (and others) so we can clean up when that happens.
$SIG{'ABRT'} = 'interruptHandler';
$SIG{'TERM'} = 'interruptHandler';
$SIG{'QUIT'} = 'interruptHandler';
$SIG{'SEGV'} = 'interruptHandler';
$SIG{'HUP' } = 'interruptHandler';
$SIG{'TRAP'} = 'interruptHandler';
$SIG{'STOP'} = 'interruptHandler';
$SIG{'INT'} = 'interruptHandler'; # handle control-c
# Set heap and stack size of krun
my $HEAP_SIZE = "EXTERN_HEAP_SIZE";
$ENV{K_OPTS} = "-Xms64m -Xmx$HEAP_SIZE -Xss32m -XX:+TieredCompilation";
my $SCRIPTS_DIR = "EXTERN_SCRIPTS_DIR";
my $EXEC_DEF = catfile($SCRIPTS_DIR, "c11-kompiled");
my $EXEC_ND_DEF = catfile($SCRIPTS_DIR, "c11-nd-kompiled");
my $EXEC_ND_THREAD_DEF = catfile($SCRIPTS_DIR, "c11-nd-thread-kompiled");
my $KRUN = "krun";
my @temporaryFiles = ();
exit main();
sub main {
my $fileInput = File::Temp->new(
TEMPLATE => 'tmp-kcc-in-XXXXXXXXXXX',
SUFFIX => '.bin',
UNLINK => 0);
my $fileOutput = File::Temp->new(
TEMPLATE => 'tmp-kcc-out-XXXXXXXXXXX',
SUFFIX => '.txt',
UNLINK => 0);
push(@temporaryFiles, $fileInput);
push(@temporaryFiles, $fileOutput);
print $fileInput linkedProgram();
my $argv = reduce { our ($a, $b); "`_List_`($a,$b)" } (map {qq|`ListItem`(#token($_, "String"))|} (map {quote(backslash(quote(backslash($_))))} ($0, @ARGV)));
my @cmd = ('--output', 'kast', '-d', $EXEC_DEF, "-cARGV=$argv", '-pARGV=printf %s', '-w', 'none', '--parser',
'cat', $fileInput, '--exit-code', "<result-value> _:Int </result-value>");
my $options = '`.Set`(.KList)';
if (defined $ENV{HELP}) {
print "Here are some configuration variables you can set to affect how this program is run:\n";
print "DEBUG --- runs krun with the --debug flag.\n";
print "TRACE --- runs krun with the --trace flag.\n";
print "DUMPALL --- leaves all the intermediate files in the current directory.\n";
print "LTLMC --- LTL model checking.\n";
print "VERBOSE --- verbose output.\n";
print "PROVE=<file> --- enable prover on specification file";
print "E.g., DEBUG=1 $0\n";
print "\n";
print "This message was displayed because the variable HELP was set. Use HELP=1 $0 to turn off.\n";
return 1;
}
if (defined $ENV{TRACE}) {
push @cmd, '--trace';
}
if (defined $ENV{DEBUG}) {
push @cmd, '--debug';
}
if (defined $ENV{VERBOSE}) {
push @cmd, '--verbose';
}
if (defined nativeLibraries()) {
push @cmd, '--native-libraries';
push @cmd, nativeLibraries();
}
if (defined nativeLibraryPath()) {
push @cmd, '--native-library-path';
push @cmd, nativeLibraryPath();
}
if (defined $ENV{PROVE}) {
push @cmd, '--prove';
push @cmd, $ENV{PROVE};
push @cmd, '--smt_prelude';
push @cmd, $ENV{SMT_PRELUDE};
} else {
push @cmd, '--smt';
push @cmd, 'none';
push @cmd, '--output-file';
push @cmd, $fileOutput;
}
if (defined $ENV{SEARCH}) {
push @cmd, '--search-final';
$options = '`SetItem`(`NOIO`(.KList))';
push @cmd, '-d';
push @cmd, $EXEC_ND_DEF;
print 'Searching reachable states... ';
print "(with non-deterministic expression sequencing)\n";
}
if (defined $ENV{THREADSEARCH}) {
push @cmd, '--search-final';
$options = '`SetItem`(`NOIO`(.KList))';
push @cmd, '-d';
push @cmd, $EXEC_ND_THREAD_DEF;
print 'Searching reachable states... ';
print "(with non-deterministic thread interleaving)\n";
}
push @cmd, "-cOPTIONS=$options";
push @cmd, '-pOPTIONS=printf %s';
# Execute krun with the arguments in @cmd
print("'krun' '" . join("' '", @cmd) . "'\n") if defined $ENV{VERBOSE};
my $ret = system($KRUN, @cmd) >> 8;
if (defined $ENV{LTLMC} | defined $ENV{PROVE}) {
return 0;
}
return processResult($fileOutput, $ret, defined $ENV{VERBOSE});
}
# TODO(chathhorn): We have two functions here because of the newlines -- the
# regexes below match against the whole file as opposed to a line at a time.
sub parseResult {
my ($parsed) = (@_);
/<k>(.*?)<\/k>/s && do {
$parsed->{finalComp} = $1;
};
/<error-cell>\s*"(.*)"\s*<\/error-cell>/ && do {
$parsed->{haveError} = 1;
my $output = $1;
$output =~ s/\%/\%\%/g;
$output =~ s/`/\\`/g;
$output =~ s/\\\\/\\\\\\\\/g;
$parsed->{errorMsg} = substr(`printf "x$output"`, 1);
};
/<curr-function>\s*(Identifier \( "(.*?)" \)|(file-scope))\s*<\/curr-function>/ && do {
$parsed->{errorFunc} = $2;
};
/<curr-program-loc>\s*CabsLoc\s*\(\s*"(.*)"\s*,\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*\)\s*<\/curr-program-loc>/ && do {
$parsed->{errorFile} = $1;
$parsed->{errorLine} = $2;
};
/<final-computation>(.*?)<\/final-computation>/s && do {
$parsed->{finalComp} = $1;
};
/<computation>(.*?)<\/computation>/s && do {
$parsed->{finalCompGoto} = $1;
};
/<type>(.*?)<\/type>/s && do {
$parsed->{finalCompType} = $1;
};
/<output>\s*#buffer\s*\(\s*"(.*)"\s*\)\s*<\/output>/ && do {
$parsed->{output} = $1;
};
/<result-value>\s*(-?\d+)\s*<\/result-value>/ && do {
$parsed->{exitCode} = $1;
};
/<result-value>\s*NullPointer\s*<\/result-value>/ && do {
$parsed->{exitCode} = 0;
};
}
sub unindent {
my ($str) = (@_);
my @lines = split "\n", $str;
for (@lines) {
s/^\s+//;
}
return join "\n", @lines;
}
sub processResult {
my ($fileOutput, $ret, $verbose) = (@_);
my %parsed;
open(OUT, "<$fileOutput");
local $/;
local $_ = <OUT>;
print if $verbose;
if ($ret) {
print "Error: Execution failed.\n";
}
return $ret;
}
sub getAttribs {
my ($nodeId, $errorStates, $goodFinal) = (@_);
my $attribs = {};
if (exists($errorStates->{$nodeId})) {
$attribs->{"fillcolor"} = "red";
$attribs->{"style"} = "filled";
}
if (exists($goodFinal->{$nodeId})) {
$attribs->{"fillcolor"} = "green";
$attribs->{"style"} = "filled";
}
return $attribs;
}
sub getString {
my ($s) = (@_);
return '' unless defined $s;
$s =~ s/\%/\%\%/g;
$s =~ s/\\\\/\\\\\\\\/g;
return substr(`printf "x$s"`, 1);
}
sub interruptHandler {
# Call single cleanup point.
finalCleanup();
kill 1, -$$;
# Since we were interrupted, we should exit with a non-zero code.
exit 1;
}
# This subroutine can be used as a way to ensure we clean up all resources
# whenever we exit. This is going to be mostly temp files. If the program
# terminates for almost any reason, this code will be executed.
sub finalCleanup {
if (!defined $ENV{DUMPALL}) {
for (@temporaryFiles) {
close $_;
unlink;
}
}
}
# This block gets run at the end of a normally terminating program, whether it
# simply exits, or dies. We use this to clean up.
END {
# $? contains the value the program would normally have exited with.
my $retval = $?;
# Call single cleanup point.
finalCleanup();
exit $retval;
}
# The parsed file contents of the program to execute with krun gets appended.