1 #!/usr/dcs/software/supported/bin/perl -w 2 # LLVM Web Demo script 3 # 4 5 use strict; 6 use CGI; 7 use POSIX; 8 use Mail::Send; 9 10 $| = 1; 11 12 my $ROOT = "/tmp/webcompile"; 13 #my $ROOT = "/home/vadve/lattner/webcompile"; 14 15 open( STDERR, ">&STDOUT" ) or die "can't redirect stderr to stdout"; 16 17 if ( !-d $ROOT ) { mkdir( $ROOT, 0777 ); } 18 19 my $LOGFILE = "$ROOT/log.txt"; 20 my $FORM_URL = 'index.cgi'; 21 my $MAILADDR = 'sabre (at] nondot.org'; 22 my $CONTACT_ADDRESS = 'Questions or comments? Email the <a href="http://lists.cs.uiuc.edu/mailman/listinfo/llvmdev">LLVMdev mailing list</a>.'; 23 my $LOGO_IMAGE_URL = 'cathead.png'; 24 my $TIMEOUTAMOUNT = 20; 25 $ENV{'LD_LIBRARY_PATH'} = '/home/vadve/shared/localtools/fc1/lib/'; 26 27 my @PREPENDPATHDIRS = 28 ( 29 '/home/vadve/shared/llvm-gcc4.0-2.1/bin/', 30 '/home/vadve/shared/llvm-2.1/Release/bin'); 31 32 my $defaultsrc = "#include <stdio.h>\n#include <stdlib.h>\n\n" . 33 "int power(int X) {\n if (X == 0) return 1;\n" . 34 " return X*power(X-1);\n}\n\n" . 35 "int main(int argc, char **argv) {\n" . 36 " printf(\"%d\\n\", power(atoi(argv[0])));\n}\n"; 37 38 sub getname { 39 my ($extension) = @_; 40 for ( my $count = 0 ; ; $count++ ) { 41 my $name = 42 sprintf( "$ROOT/_%d_%d%s", $$, $count, $extension ); 43 if ( !-f $name ) { return $name; } 44 } 45 } 46 47 my $c; 48 49 sub barf { 50 print "<b>", @_, "</b>\n"; 51 print $c->end_html; 52 system("rm -f $ROOT/locked"); 53 exit 1; 54 } 55 56 sub writeIntoFile { 57 my $extension = shift @_; 58 my $contents = join "", @_; 59 my $name = getname($extension); 60 local (*FILE); 61 open( FILE, ">$name" ) or barf("Can't write to $name: $!"); 62 print FILE $contents; 63 close FILE; 64 return $name; 65 } 66 67 sub addlog { 68 my ( $source, $pid, $result ) = @_; 69 open( LOG, ">>$LOGFILE" ); 70 my $time = scalar localtime; 71 my $remotehost = $ENV{'REMOTE_ADDR'}; 72 print LOG "[$time] [$remotehost]: $pid\n"; 73 print LOG "<<<\n$source\n>>>\nResult is: <<<\n$result\n>>>\n"; 74 close LOG; 75 } 76 77 sub dumpFile { 78 my ( $header, $file ) = @_; 79 my $result; 80 open( FILE, "$file" ) or barf("Can't read $file: $!"); 81 while (<FILE>) { 82 $result .= $_; 83 } 84 close FILE; 85 my $UnhilightedResult = $result; 86 my $HtmlResult = 87 "<h3>$header</h3>\n<pre>\n" . $c->escapeHTML($result) . "\n</pre>\n"; 88 if (wantarray) { 89 return ( $UnhilightedResult, $HtmlResult ); 90 } 91 else { 92 return $HtmlResult; 93 } 94 } 95 96 sub syntaxHighlightLLVM { 97 my ($input) = @_; 98 $input =~ s@\b(void|i8|i1|i16|i32|i64|float|double|type|label|opaque)\b@<span class="llvm_type">$1</span>@g; 99 $input =~ s@\b(add|sub|mul|div|rem|and|or|xor|setne|seteq|setlt|setgt|setle|setge|phi|tail|call|cast|to|shl|shr|vaarg|vanext|ret|br|switch|invoke|unwind|malloc|alloca|free|load|store|getelementptr|begin|end|true|false|declare|global|constant|const|internal|uninitialized|external|implementation|linkonce|weak|appending|null|to|except|not|target|endian|pointersize|big|little|volatile)\b@<span class="llvm_keyword">$1</span>@g; 100 101 # Add links to the FAQ. 102 $input =~ s@(_ZNSt8ios_base4Init[DC]1Ev)@<a href="../docs/FAQ.html#iosinit">$1</a>@g; 103 $input =~ s@\bundef\b@<a href="../docs/FAQ.html#undef">undef</a>@g; 104 return $input; 105 } 106 107 sub mailto { 108 my ( $recipient, $body ) = @_; 109 my $msg = 110 new Mail::Send( Subject => "LLVM Demo Page Run", To => $recipient ); 111 my $fh = $msg->open(); 112 print $fh $body; 113 $fh->close(); 114 } 115 116 $c = new CGI; 117 print $c->header; 118 119 print <<EOF; 120 <html> 121 <head> 122 <meta http-equiv="Content-Type" content="text/html; charset=iso-8859-1"> 123 <title>Try out LLVM in your browser!</title> 124 <style> 125 \@import url("syntax.css"); 126 \@import url("http://llvm.org/llvm.css"); 127 </style> 128 </head> 129 <body leftmargin="10" marginwidth="10"> 130 131 <div class="www_sectiontitle"> 132 Try out LLVM in your browser! 133 </div> 134 135 <table border=0><tr><td> 136 <img align=right width=100 height=111 src="$LOGO_IMAGE_URL"> 137 </td><td> 138 EOF 139 140 if ( -f "$ROOT/locked" ) { 141 my ($dev,$ino,$mode,$nlink,$uid,$gid,$rdev,$size,$atime,$locktime) = 142 stat("$ROOT/locked"); 143 my $currtime = time(); 144 if ($locktime + 60 > $currtime) { 145 print "This page is already in use by someone else at this "; 146 print "time, try reloading in a second or two. Meow!</td></tr></table>'\n"; 147 exit 0; 148 } 149 } 150 151 system("touch $ROOT/locked"); 152 153 print <<END; 154 Bitter Melon the cat says, paste a C/C++ program in the text box or upload 155 one from your computer, and you can see LLVM compile it, meow!! 156 </td></tr></table><p> 157 END 158 159 print $c->start_multipart_form( 'POST', $FORM_URL ); 160 161 my $source = $c->param('source'); 162 163 164 # Start the user out with something valid if no code. 165 $source = $defaultsrc if (!defined($source)); 166 167 print '<table border="0"><tr><td>'; 168 169 print "Type your source code in below: (<a href='DemoInfo.html#hints'>hints and 170 advice</a>)<br>\n"; 171 172 print $c->textarea( 173 -name => "source", 174 -rows => 16, 175 -columns => 60, 176 -default => $source 177 ), "<br>"; 178 179 print "Or upload a file: "; 180 print $c->filefield( -name => 'uploaded_file', -default => '' ); 181 182 print "<p />\n"; 183 184 185 print '<p></td><td valign=top>'; 186 187 print "<center><h3>General Options</h3></center>"; 188 189 print "Source language: ", 190 $c->radio_group( 191 -name => 'language', 192 -values => [ 'C', 'C++' ], 193 -default => 'C' 194 ), "<p>"; 195 196 print $c->checkbox( 197 -name => 'linkopt', 198 -label => 'Run link-time optimizer', 199 -checked => 'checked' 200 ),' <a href="DemoInfo.html#lto">?</a><br>'; 201 202 print $c->checkbox( 203 -name => 'showstats', 204 -label => 'Show detailed pass statistics' 205 ), ' <a href="DemoInfo.html#stats">?</a><br>'; 206 207 print $c->checkbox( 208 -name => 'cxxdemangle', 209 -label => 'Demangle C++ names' 210 ),' <a href="DemoInfo.html#demangle">?</a><p>'; 211 212 213 print "<center><h3>Output Options</h3></center>"; 214 215 print $c->checkbox( 216 -name => 'showbcanalysis', 217 -label => 'Show detailed bytecode analysis' 218 ),' <a href="DemoInfo.html#bcanalyzer">?</a><br>'; 219 220 print $c->checkbox( 221 -name => 'showllvm2cpp', 222 -label => 'Show LLVM C++ API code' 223 ), ' <a href="DemoInfo.html#llvm2cpp">?</a>'; 224 225 print "</td></tr></table>"; 226 227 print "<center>", $c->submit(-value=> 'Compile Source Code'), 228 "</center>\n", $c->endform; 229 230 print "\n<p>If you have questions about the LLVM code generated by the 231 front-end, please check the <a href='/docs/FAQ.html#cfe_code'>FAQ</a> and 232 the demo page <a href='DemoInfo.html#hints'>hints section</a>. 233 </p>\n"; 234 235 $ENV{'PATH'} = ( join ( ':', @PREPENDPATHDIRS ) ) . ":" . $ENV{'PATH'}; 236 237 sub sanitychecktools { 238 my $sanitycheckfail = ''; 239 240 # insert tool-specific sanity checks here 241 $sanitycheckfail .= ' llvm-dis' 242 if `llvm-dis --help 2>&1` !~ /ll disassembler/; 243 244 $sanitycheckfail .= ' llvm-gcc' 245 if ( `llvm-gcc --version 2>&1` !~ /Free Software Foundation/ ); 246 247 $sanitycheckfail .= ' llvm-ld' 248 if `llvm-ld --help 2>&1` !~ /llvm linker/; 249 250 $sanitycheckfail .= ' llvm-bcanalyzer' 251 if `llvm-bcanalyzer --help 2>&1` !~ /bcanalyzer/; 252 253 barf( 254 "<br/>The demo page is currently unavailable. [tools: ($sanitycheckfail ) failed sanity check]" 255 ) 256 if $sanitycheckfail; 257 } 258 259 sanitychecktools(); 260 261 sub try_run { 262 my ( $program, $commandline, $outputFile ) = @_; 263 my $retcode = 0; 264 265 eval { 266 local $SIG{ALRM} = sub { die "timeout"; }; 267 alarm $TIMEOUTAMOUNT; 268 $retcode = system($commandline); 269 alarm 0; 270 }; 271 if ( $@ and $@ =~ /timeout/ ) { 272 barf("Program $program took too long, compile time limited for the web script, sorry!\n"); 273 } 274 if ( -s $outputFile ) { 275 print scalar dumpFile( "Output from $program", $outputFile ); 276 } 277 #print "<p>Finished dumping command output.</p>\n"; 278 if ( WIFEXITED($retcode) && WEXITSTATUS($retcode) != 0 ) { 279 barf( 280 "$program exited with an error. Please correct source and resubmit.<p>\n" . 281 "Please note that this form only allows fully formed and correct source" . 282 " files. It will not compile fragments of code.<p>" 283 ); 284 } 285 if ( WIFSIGNALED($retcode) != 0 ) { 286 my $sig = WTERMSIG($retcode); 287 barf( 288 "Ouch, $program caught signal $sig. Sorry, better luck next time!\n" 289 ); 290 } 291 } 292 293 my %suffixes = ( 294 'Java' => '.java', 295 'JO99' => '.jo9', 296 'C' => '.c', 297 'C++' => '.cc', 298 'Stacker' => '.st', 299 'preprocessed C' => '.i', 300 'preprocessed C++' => '.ii' 301 ); 302 my %languages = ( 303 '.jo9' => 'JO99', 304 '.java' => 'Java', 305 '.c' => 'C', 306 '.i' => 'preprocessed C', 307 '.ii' => 'preprocessed C++', 308 '.cc' => 'C++', 309 '.cpp' => 'C++', 310 '.st' => 'Stacker' 311 ); 312 313 my $uploaded_file_name = $c->param('uploaded_file'); 314 if ($uploaded_file_name) { 315 if ($source) { 316 barf( 317 "You must choose between uploading a file and typing code in. You can't do both at the same time." 318 ); 319 } 320 $uploaded_file_name =~ s/^.*(\.[A-Za-z]+)$/$1/; 321 my $language = $languages{$uploaded_file_name}; 322 $c->param( 'language', $language ); 323 324 print "<p>Processing uploaded file. It looks like $language.</p>\n"; 325 my $fh = $c->upload('uploaded_file'); 326 if ( !$fh ) { 327 barf( "Error uploading file: " . $c->cgi_error ); 328 } 329 while (<$fh>) { 330 $source .= $_; 331 } 332 close $fh; 333 } 334 335 if ($c->param('source')) { 336 print $c->hr; 337 my $extension = $suffixes{ $c->param('language') }; 338 barf "Unknown language; can't compile\n" unless $extension; 339 340 # Add a newline to the source here to avoid a warning from gcc. 341 $source .= "\n"; 342 343 # Avoid security hole due to #including bad stuff. 344 $source =~ 345 s@(\n)?#include.*[<"](.*\.\..*)[">].*\n@$1#error "invalid #include file $2 detected"\n@g; 346 347 my $inputFile = writeIntoFile( $extension, $source ); 348 my $pid = $$; 349 350 my $bytecodeFile = getname(".bc"); 351 my $outputFile = getname(".llvm-gcc.out"); 352 my $timerFile = getname(".llvm-gcc.time"); 353 354 my $stats = ''; 355 if ( $extension eq ".st" ) { 356 $stats = "-stats -time-passes " 357 if ( $c->param('showstats') ); 358 try_run( "llvm Stacker front-end (stkrc)", 359 "stkrc $stats -o $bytecodeFile $inputFile > $outputFile 2>&1", 360 $outputFile ); 361 } else { 362 #$stats = "-Wa,--stats,--time-passes,--info-output-file=$timerFile" 363 $stats = "-ftime-report" 364 if ( $c->param('showstats') ); 365 try_run( "llvm C/C++ front-end (llvm-gcc)", 366 "llvm-gcc -emit-llvm -W -Wall -O2 $stats -o $bytecodeFile -c $inputFile > $outputFile 2>&1", 367 $outputFile ); 368 } 369 370 if ( $c->param('showstats') && -s $timerFile ) { 371 my ( $UnhilightedResult, $HtmlResult ) = 372 dumpFile( "Statistics for front-end compilation", $timerFile ); 373 print "$HtmlResult\n"; 374 } 375 376 if ( $c->param('linkopt') ) { 377 my $stats = ''; 378 my $outputFile = getname(".gccld.out"); 379 my $timerFile = getname(".gccld.time"); 380 $stats = "--stats --time-passes --info-output-file=$timerFile" 381 if ( $c->param('showstats') ); 382 my $tmpFile = getname(".bc"); 383 try_run( 384 "optimizing linker (llvm-ld)", 385 "llvm-ld $stats -o=$tmpFile $bytecodeFile > $outputFile 2>&1", 386 $outputFile 387 ); 388 system("mv $tmpFile.bc $bytecodeFile"); 389 system("rm $tmpFile"); 390 391 if ( $c->param('showstats') && -s $timerFile ) { 392 my ( $UnhilightedResult, $HtmlResult ) = 393 dumpFile( "Statistics for optimizing linker", $timerFile ); 394 print "$HtmlResult\n"; 395 } 396 } 397 398 print " Bytecode size is ", -s $bytecodeFile, " bytes.\n"; 399 400 my $disassemblyFile = getname(".ll"); 401 try_run( "llvm-dis", 402 "llvm-dis -o=$disassemblyFile $bytecodeFile > $outputFile 2>&1", 403 $outputFile ); 404 405 if ( $c->param('cxxdemangle') ) { 406 print " Demangling disassembler output.\n"; 407 my $tmpFile = getname(".ll"); 408 system("c++filt < $disassemblyFile > $tmpFile 2>&1"); 409 system("mv $tmpFile $disassemblyFile"); 410 } 411 412 my ( $UnhilightedResult, $HtmlResult ); 413 if ( -s $disassemblyFile ) { 414 ( $UnhilightedResult, $HtmlResult ) = 415 dumpFile( "Output from LLVM disassembler", $disassemblyFile ); 416 print syntaxHighlightLLVM($HtmlResult); 417 } 418 else { 419 print "<p>Hmm, that's weird, llvm-dis didn't produce any output.</p>\n"; 420 } 421 422 if ( $c->param('showbcanalysis') ) { 423 my $analFile = getname(".bca"); 424 try_run( "llvm-bcanalyzer", "llvm-bcanalyzer $bytecodeFile > $analFile 2>&1", 425 $analFile); 426 } 427 if ($c->param('showllvm2cpp') ) { 428 my $l2cppFile = getname(".l2cpp"); 429 try_run("llvm2cpp","llvm2cpp $bytecodeFile -o $l2cppFile 2>&1", 430 $l2cppFile); 431 } 432 433 # Get the source presented by the user to CGI, convert newline sequences to simple \n. 434 my $actualsrc = $c->param('source'); 435 $actualsrc =~ s/\015\012/\n/go; 436 # Don't log this or mail it if it is the default code. 437 if ($actualsrc ne $defaultsrc) { 438 addlog( $source, $pid, $UnhilightedResult ); 439 440 my ( $ip, $host, $lg, $lines ); 441 chomp( $lines = `wc -l < $inputFile` ); 442 $lg = $c->param('language'); 443 $ip = $c->remote_addr(); 444 chomp( $host = `host $ip` ) if $ip; 445 mailto( $MAILADDR, 446 "--- Query: ---\nFrom: ($ip) $host\nInput: $lines lines of $lg\n" 447 . "C++ demangle = " 448 . ( $c->param('cxxdemangle') ? 1 : 0 ) 449 . ", Link opt = " 450 . ( $c->param('linkopt') ? 1 : 0 ) . "\n\n" 451 . ", Show stats = " 452 . ( $c->param('showstats') ? 1 : 0 ) . "\n\n" 453 . "--- Source: ---\n$source\n" 454 . "--- Result: ---\n$UnhilightedResult\n" ); 455 } 456 unlink( $inputFile, $bytecodeFile, $outputFile, $disassemblyFile ); 457 } 458 459 print $c->hr, "<address>$CONTACT_ADDRESS</address>", $c->end_html; 460 system("rm $ROOT/locked"); 461 exit 0; 462