comparison graal/com.oracle.max.base/src/com/sun/max/test/JavaExecHarness.java @ 3733:e233f5660da4

Added Java files from Maxine project.
author Thomas Wuerthinger <thomas.wuerthinger@oracle.com>
date Sat, 17 Dec 2011 19:59:18 +0100
parents
children bc8527f3071c
comparison
equal deleted inserted replaced
3732:3e2e8b8abdaf 3733:e233f5660da4
1 /*
2 * Copyright (c) 2007, 2011, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.
8 *
9 * This code is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12 * version 2 for more details (a copy is included in the LICENSE file that
13 * accompanied this code).
14 *
15 * You should have received a copy of the GNU General Public License version
16 * 2 along with this work; if not, write to the Free Software Foundation,
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20 * or visit www.oracle.com if you need additional information or have any
21 * questions.
22 */
23 package com.sun.max.test;
24
25 import java.io.*;
26 import java.lang.reflect.*;
27 import java.text.*;
28 import java.util.*;
29
30 import com.sun.max.lang.*;
31 import com.sun.max.program.*;
32
33 public class JavaExecHarness implements TestHarness<JavaExecHarness.JavaTestCase> {
34
35 private static final char SQUOTE = '\'';
36 private static final char BACKSLASH = '\\';
37 private static final char QUOTE = '"';
38 private static final String ESCAPED_QUOTE = "\\\"";
39
40 private final Executor executor;
41
42 public class CodeLiteral {
43 public String codeLiteral;
44 CodeLiteral(String codeLiteral) {
45 this.codeLiteral = codeLiteral;
46 }
47 @Override
48 public String toString() {
49 return codeLiteral;
50 }
51
52 @Override
53 public boolean equals(Object obj) {
54 if (!(obj instanceof CodeLiteral)) {
55 return resolve().equals(obj);
56 }
57 return super.equals(obj);
58 }
59
60 public Object resolve() {
61 String s = codeLiteral;
62 String className = s.substring(0, s.lastIndexOf('.'));
63 String fieldName = s.substring(s.lastIndexOf('.') + 1);
64 Class klass;
65 try {
66 klass = Class.forName(className);
67 return klass.getField(fieldName).get(null);
68 } catch (ClassNotFoundException e) {
69 throw new RuntimeException(e);
70 } catch (IllegalArgumentException e) {
71 throw new RuntimeException(e);
72 } catch (SecurityException e) {
73 throw new RuntimeException(e);
74 } catch (IllegalAccessException e) {
75 throw new RuntimeException(e);
76 } catch (NoSuchFieldException e) {
77 throw new RuntimeException(e);
78 }
79 }
80 }
81
82 public class MethodCall extends CodeLiteral {
83 MethodCall(String codeCall) {
84 super(codeCall);
85 }
86 }
87
88 public interface Executor {
89 void initialize(JavaTestCase testCase, boolean loadingPackages);
90
91 Object execute(JavaTestCase c, Object[] vals) throws InvocationTargetException;
92 }
93
94 public JavaExecHarness(Executor exec) {
95 executor = exec;
96 }
97
98 public static class Run {
99 public final Object[] input;
100 public final Object expectedValue;
101 public final Class<? extends Throwable> expectedException;
102
103 Object returnVal;
104 Throwable returnExc;
105 Throwable thrown;
106
107 Run(Object[] input, Object expected, Class<? extends Throwable> expectedException) {
108 this.input = input;
109 expectedValue = expected;
110 this.expectedException = expectedException;
111 }
112 }
113
114 public class JavaTestCase extends TestCase {
115 public final Class clazz;
116 public final List<Run> runs;
117 public final Executor exec;
118 public final String execName;
119
120 public Object slot1;
121 public Object slot2;
122
123 protected JavaTestCase(String execName, Executor executor, File file, Class testClass, List<Run> runs, boolean loadingPackages) {
124 super(JavaExecHarness.this, file, null);
125 this.execName = execName;
126 exec = executor;
127 this.runs = runs;
128 clazz = testClass;
129 executor.initialize(this, loadingPackages);
130 }
131
132 @Override
133 public void run() throws Throwable {
134 for (Run run : runs) {
135 try {
136 run.returnVal = exec.execute(this, run.input);
137 } catch (InvocationTargetException t) {
138 run.returnExc = t.getTargetException();
139 } catch (Throwable t) {
140 run.thrown = t;
141 }
142 }
143 }
144 }
145
146 public static class ExecFailure extends TestResult.Failure {
147 protected final Run run;
148 protected final String result;
149 protected final String expect;
150 protected ExecFailure(Run run, String result) {
151 this.run = run;
152 this.expect = resultToString(run.expectedValue, run.expectedException);
153 this.result = result;
154 }
155 @Override
156 public String failureMessage(TestCase testCase) {
157 final JavaTestCase javaTestCase = (JavaTestCase) testCase;
158 return inputToString(javaTestCase.clazz, run, false) + " failed with " + result + " (expected " + expect + ")";
159 }
160
161 }
162
163 public TestResult evaluateTest(TestEngine engine, JavaTestCase testCase) {
164 if (testCase.thrown != null) {
165 return new TestResult.UnexpectedException(testCase.thrown);
166 }
167 for (Run run : testCase.runs) {
168 if (run.thrown != null) {
169 return new ExecFailure(run, "unexpected " + run.thrown.getClass().getName() + " (\"" + run.thrown.getMessage() + "\")");
170 }
171 final String result = valueToString(run.returnVal, run.returnExc);
172 if (run.expectedException != null) {
173 if (run.returnExc == null || run.returnExc.getClass() != run.expectedException) {
174 return new ExecFailure(run, result);
175 }
176 } else if (run.returnExc != null) {
177 return new ExecFailure(run, result);
178 } else if (run.expectedValue == null) {
179 if (run.returnVal != null) {
180 return new ExecFailure(run, result);
181 }
182 } else if (!run.expectedValue.equals(run.returnVal)) {
183 return new ExecFailure(run, result);
184 }
185 }
186 return TestResult.SUCCESS;
187 }
188
189 public void parseTests(TestEngine engine, File file, Properties props) {
190 try {
191 // 1. find the class
192 final Class testClass = findClass(file, props);
193 // 2. parse the runs
194 final List<Run> runs = parseRuns(testClass, file, props);
195 if (runs != null) {
196 // 3. add a test case to the engine
197 engine.addTest(new JavaTestCase("exec", executor, file, testClass, runs, engine.loadingPackages()));
198 } else {
199 engine.skipFile(file);
200 }
201 } catch (Exception e1) {
202 throw ProgramError.unexpected(e1);
203 }
204 }
205
206 private Class findClass(File file, Properties props) throws Exception {
207 final BufferedReader r = new BufferedReader(new FileReader(file));
208
209 // search for the package statement in the file.
210 String line = r.readLine();
211 for (; line != null; line = r.readLine()) {
212 line = line.trim();
213 if (line.startsWith("package")) {
214 // this is probably a java file
215 r.close();
216 int indx = line.indexOf(' ');
217 while (line.charAt(indx) == ' ') {
218 indx++;
219 }
220 final String packageName = line.substring(indx, line.indexOf(';'));
221 String className = file.getName();
222 if (className.endsWith(".java")) {
223 className = className.substring(0, className.length() - ".java".length());
224 }
225 // use the package name plus the name of the file to load the class.
226 return Class.forName(packageName + "." + className);
227 }
228 if (line.startsWith(".class")) {
229 // this is probably a jasm file
230 String[] tokens = line.split(" ");
231 String className = null;
232 for (String s : tokens) {
233 if (!".class".equals(s) && !"public".equals(s) && !"abstract".equals(s)) {
234 className = s;
235 break;
236 }
237 }
238 return Class.forName(className.replace('/', '.'));
239 }
240 }
241 r.close();
242 throw ProgramError.unexpected("could not find package statement in " + file);
243 }
244
245 private List<Run> parseRuns(Class testClass, File file, Properties props) {
246 final String rstr = props.getProperty("Runs");
247 if (rstr == null) {
248 return null;
249 }
250 final List<Run> runs = new LinkedList<Run>();
251 final CharacterIterator i = new StringCharacterIterator(rstr);
252 while (i.getIndex() < i.getEndIndex()) {
253 runs.add(parseRun(i));
254 if (!skipPeekAndEat(i, ';')) {
255 break;
256 }
257 }
258 return runs;
259 }
260
261 private Run parseRun(CharacterIterator iterator) {
262 // parses strings of the form:
263 // ()=value
264 // (value,...)=result
265 // value=result
266 Object[] vals = new Object[1];
267 if (skipPeekAndEat(iterator, '(')) {
268 final List<Object> inputValues = new LinkedList<Object>();
269 if (!skipPeekAndEat(iterator, ')')) {
270 while (true) {
271 inputValues.add(parseValue(iterator));
272 if (!skipPeekAndEat(iterator, ',')) {
273 break;
274 }
275 }
276 skipPeekAndEat(iterator, ')');
277 }
278 vals = inputValues.toArray(vals);
279 } else {
280 vals[0] = parseValue(iterator);
281 }
282 skipPeekAndEat(iterator, '=');
283 if (skipPeekAndEat(iterator, '!')) {
284 return new Run(vals, null, parseException(iterator));
285 }
286 return new Run(vals, parseValue(iterator), null);
287 }
288
289 private Object parseValue(CharacterIterator iterator) {
290 // parses strings of the form:
291 // <integer> | <long> | <string> | true | false | null
292 skipWhitespace(iterator);
293 if (iterator.current() == '-' || Character.isDigit(iterator.current())) {
294 // parse a number.
295 return parseNumber(iterator);
296 } else if (iterator.current() == QUOTE) {
297 // a string constant.
298 return parseStringLiteral(iterator);
299 } else if (peekAndEat(iterator, "true")) {
300 // the boolean value (true)
301 return Boolean.TRUE;
302 } else if (peekAndEat(iterator, "false")) {
303 // the boolean value (false)
304 return Boolean.FALSE;
305 } else if (peekAndEat(iterator, "null")) {
306 // the null value (null)
307 return null;
308 } else if (iterator.current() == '`') {
309 expectChar(iterator, '`');
310 return new CodeLiteral(parseCodeLiteral(iterator));
311 } else if (iterator.current() == '(') {
312 expectChar(iterator, '(');
313 expectChar(iterator, ')');
314 return new MethodCall(parseCodeLiteral(iterator));
315 }
316 throw ProgramError.unexpected("invalid value at " + iterator.getIndex());
317 }
318
319 private ProgramError raiseParseErrorAt(String message, CharacterIterator iterator) {
320 final int errorIndex = iterator.getIndex();
321 final StringBuilder sb = new StringBuilder(message).append(String.format(":%n"));
322 iterator.setIndex(iterator.getBeginIndex());
323 for (char ch = iterator.current(); ch != CharacterIterator.DONE; ch = iterator.next()) {
324 sb.append(ch);
325 }
326 sb.append(String.format("%n"));
327 for (int i = 0; i < errorIndex; ++i) {
328 sb.append(' ');
329 }
330 sb.append('^');
331 throw ProgramError.unexpected(sb.toString());
332 }
333
334 private Object parseNumber(CharacterIterator iterator) {
335 // an integer.
336 final StringBuilder buf = new StringBuilder();
337
338 if (iterator.current() == '-') {
339 buf.append('-');
340 iterator.next();
341 }
342
343 int radix = 10;
344 if (iterator.current() == '0') {
345 radix = 8;
346 iterator.next();
347 if (iterator.current() == 'x' || iterator.current() == 'X') {
348 radix = 16;
349 iterator.next();
350 } else if (Character.digit(iterator.current(), 8) == -1) {
351 radix = 10;
352 buf.append('0');
353 }
354 }
355 appendDigits(buf, iterator, radix);
356
357 if (peekAndEat(iterator, '.')) {
358 if (radix != 10) {
359 raiseParseErrorAt("Cannot have decimal point in number with radix " + radix, iterator);
360 }
361 // parse the fractional suffix of a float or double
362 buf.append('.');
363 appendDigits(buf, iterator, radix);
364 if (peekAndEat(iterator, 'f') || peekAndEat(iterator, "F")) {
365 return Float.valueOf(buf.toString());
366 }
367 if (peekAndEat(iterator, 'd') || peekAndEat(iterator, "D")) {
368 return Double.valueOf(buf.toString());
369 }
370 return Float.valueOf(buf.toString());
371 }
372 if (radix == 10) {
373 if (peekAndEat(iterator, 'f') || peekAndEat(iterator, "F")) {
374 return Float.valueOf(buf.toString());
375 }
376 if (peekAndEat(iterator, 'd') || peekAndEat(iterator, "D")) {
377 return Double.valueOf(buf.toString());
378 }
379 if (peekAndEat(iterator, 's') || peekAndEat(iterator, "S")) {
380 return Short.valueOf(buf.toString());
381 }
382 if (peekAndEat(iterator, 'b') || peekAndEat(iterator, "B")) {
383 return Byte.valueOf(buf.toString());
384 }
385 if (peekAndEat(iterator, 'c') || peekAndEat(iterator, "C")) {
386 return Character.valueOf((char) Integer.valueOf(buf.toString()).intValue());
387 }
388 }
389 if (peekAndEat(iterator, 'l') || peekAndEat(iterator, "L")) {
390 return Long.valueOf(buf.toString(), radix);
391 }
392 return Integer.valueOf(buf.toString(), radix);
393 }
394
395 private void appendDigits(final StringBuilder buf, CharacterIterator iterator, int radix) {
396 while (Character.digit(iterator.current(), radix) != -1) {
397 buf.append(iterator.current());
398 iterator.next();
399 }
400 }
401
402 private Class<? extends Throwable> parseException(CharacterIterator iterator) {
403 final String exceptionName = parseCodeLiteral(iterator);
404 try {
405 return Class.forName(exceptionName).asSubclass(Throwable.class);
406 } catch (ClassNotFoundException e) {
407 throw raiseParseErrorAt("Unknown exception type", iterator);
408 }
409 }
410
411 private String parseCodeLiteral(CharacterIterator iterator) {
412 final StringBuilder buf = new StringBuilder();
413 while (true) {
414 final char ch = iterator.current();
415 if (Character.isJavaIdentifierPart(ch) || ch == '.') {
416 buf.append(ch);
417 iterator.next();
418 } else {
419 break;
420 }
421 }
422 return buf.toString();
423 }
424
425 private boolean skipPeekAndEat(CharacterIterator iterator, char c) {
426 skipWhitespace(iterator);
427 return peekAndEat(iterator, c);
428 }
429
430 private boolean peekAndEat(CharacterIterator iterator, char c) {
431 if (iterator.current() == c) {
432 iterator.next();
433 return true;
434 }
435 return false;
436 }
437
438 private boolean peekAndEat(CharacterIterator iterator, String string) {
439 final int indx = iterator.getIndex();
440 for (int j = 0; j < string.length(); j++) {
441 if (iterator.current() != string.charAt(j)) {
442 iterator.setIndex(indx);
443 return false;
444 }
445 iterator.next();
446 }
447 return true;
448 }
449
450 private void skipWhitespace(CharacterIterator iterator) {
451 while (true) {
452 if (!Character.isWhitespace(iterator.current())) {
453 break;
454 }
455 iterator.next();
456 }
457 }
458
459 private void expectChar(CharacterIterator i, char c) {
460 final char r = i.current();
461 i.next();
462 if (r != c) {
463 throw ProgramError.unexpected("parse error at " + i.getIndex() + ", expected character '" + c + "'");
464 }
465 }
466
467 private char parseCharLiteral(CharacterIterator i) throws Exception {
468
469 expectChar(i, SQUOTE);
470
471 char ch;
472 if (peekAndEat(i, BACKSLASH)) {
473 ch = parseEscapeChar(i);
474 } else {
475 ch = i.current();
476 i.next();
477 }
478
479 expectChar(i, SQUOTE);
480
481 return ch;
482 }
483
484 private char parseEscapeChar(CharacterIterator i) {
485 final char c = i.current();
486 switch (c) {
487 case 'f':
488 i.next();
489 return '\f';
490 case 'b':
491 i.next();
492 return '\b';
493 case 'n':
494 i.next();
495 return '\n';
496 case 'r':
497 i.next();
498 return '\r';
499 case BACKSLASH:
500 i.next();
501 return BACKSLASH;
502 case SQUOTE:
503 i.next();
504 return SQUOTE;
505 case QUOTE:
506 i.next();
507 return QUOTE;
508 case 't':
509 i.next();
510 return '\t';
511 case 'x':
512 return (char) readHexValue(i, 4);
513 case '0': // fall through
514 case '1': // fall through
515 case '2': // fall through
516 case '3': // fall through
517 case '4': // fall through
518 case '5': // fall through
519 case '6': // fall through
520 case '7':
521 return (char) readOctalValue(i, 3);
522
523 }
524 return c;
525 }
526
527 private String parseStringLiteral(CharacterIterator i) {
528 final StringBuilder buffer = new StringBuilder(i.getEndIndex() - i.getBeginIndex() + 1);
529
530 expectChar(i, QUOTE);
531 while (true) {
532 if (peekAndEat(i, QUOTE)) {
533 break;
534 }
535 char c = i.current();
536 i.next();
537
538 if (c == CharacterIterator.DONE) {
539 break;
540 }
541 if (c == BACKSLASH) {
542 c = parseEscapeChar(i);
543 }
544
545 buffer.append(c);
546 }
547
548 return buffer.toString();
549 }
550
551 public static int readHexValue(CharacterIterator i, int maxchars) {
552 int accumul = 0;
553
554 for (int cntr = 0; cntr < maxchars; cntr++) {
555 final char c = i.current();
556
557 if (c == CharacterIterator.DONE || !Chars.isHexDigit(c)) {
558 break;
559 }
560
561 accumul = (accumul << 4) | Character.digit(c, 16);
562 i.next();
563 }
564
565 return accumul;
566 }
567
568 public static int readOctalValue(CharacterIterator i, int maxchars) {
569 int accumul = 0;
570
571 for (int cntr = 0; cntr < maxchars; cntr++) {
572 final char c = i.current();
573
574 if (!Chars.isOctalDigit(c)) {
575 break;
576 }
577
578 accumul = (accumul << 3) | Character.digit(c, 8);
579 i.next();
580 }
581
582 return accumul;
583 }
584
585 public static String inputToString(Class testClass, Run run, boolean asJavaString) {
586 final StringBuilder buffer = new StringBuilder();
587 if (asJavaString) {
588 buffer.append(QUOTE);
589 }
590 buffer.append("(");
591 for (int i = 0; i < run.input.length; i++) {
592 if (i > 0) {
593 buffer.append(',');
594 }
595 final Object val = run.input[i];
596 if (val instanceof Character) {
597 buffer.append(Chars.toJavaLiteral((Character) val));
598 } else if (val instanceof String) {
599 buffer.append(asJavaString ? ESCAPED_QUOTE : QUOTE);
600 buffer.append(val);
601 buffer.append(asJavaString ? ESCAPED_QUOTE : QUOTE);
602 } else {
603 buffer.append(String.valueOf(val));
604 }
605 }
606 buffer.append(')');
607 if (asJavaString) {
608 buffer.append(QUOTE);
609 }
610 return buffer.toString();
611 }
612
613 public static String resultToString(Object val, Class<? extends Throwable> throwable) {
614 if (throwable != null) {
615 return "!" + throwable.getName();
616 }
617 if (val instanceof Character) {
618 return Chars.toJavaLiteral((Character) val);
619 }
620 if (val instanceof String) {
621 return "\"" + val + "\"";
622 }
623 return String.valueOf(val);
624 }
625
626 public static String valueToString(Object val, Throwable thrown) {
627 if (thrown == null) {
628 return resultToString(val, null);
629 }
630 return resultToString(val, thrown.getClass()) + "(" + thrown.getMessage() + ")";
631 }
632 }