|
| 1 | +package edu.columbia.cs.psl.chroniclerj; |
| 2 | + |
| 3 | +import java.lang.instrument.ClassFileTransformer; |
| 4 | +import java.lang.instrument.IllegalClassFormatException; |
| 5 | +import java.lang.instrument.Instrumentation; |
| 6 | +import java.security.ProtectionDomain; |
| 7 | + |
| 8 | +import org.objectweb.asm.ClassReader; |
| 9 | +import org.objectweb.asm.ClassWriter; |
| 10 | +import org.objectweb.asm.Opcodes; |
| 11 | +import org.objectweb.asm.commons.SerialVersionUIDAdder; |
| 12 | + |
| 13 | +import edu.columbia.cs.psl.chroniclerj.replay.NonDeterministicReplayClassVisitor; |
| 14 | +import edu.columbia.cs.psl.chroniclerj.visitor.CallbackDuplicatingClassVisitor; |
| 15 | +import edu.columbia.cs.psl.chroniclerj.visitor.NonDeterministicLoggingClassVisitor; |
| 16 | + |
| 17 | +public class PreMain { |
| 18 | + public static boolean isIgnoredClass(String className) { |
| 19 | + return className.startsWith("java") || className.startsWith("com/sun") || className.startsWith("sun/") || className.startsWith("edu/columbia/cs/psl/chroniclerj") || className.startsWith("com/rits/cloning") || className.startsWith("jdk") |
| 20 | + ||className.startsWith("com/thoughtworks") || className.startsWith("org/xmlpull") |
| 21 | + || className.startsWith("org/kxml2"); |
| 22 | + } |
| 23 | + |
| 24 | + public static void premain(String _args, Instrumentation inst) { |
| 25 | + final boolean replay = _args != null && _args.equals("replay"); |
| 26 | + if(_args != null) |
| 27 | + { |
| 28 | + String[] args = _args.split(","); |
| 29 | + for(String arg : args) |
| 30 | + { |
| 31 | + String[] d = arg.split("="); |
| 32 | + if(d[0].equals("logFile")) |
| 33 | + { |
| 34 | + ChroniclerJExportRunner.nameOverride = d[1]; |
| 35 | + } |
| 36 | + else if(d[0].equals("alwaysExport")) |
| 37 | + { |
| 38 | + ChroniclerJExportRunner.registerShutdownHook(); |
| 39 | + } |
| 40 | + } |
| 41 | + } |
| 42 | + ClassFileTransformer transformer = new ClassFileTransformer() { |
| 43 | + |
| 44 | + @Override |
| 45 | + public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException { |
| 46 | + if (replay) { |
| 47 | + ClassReader cr = new ClassReader(classfileBuffer); |
| 48 | + if (isIgnoredClass(cr.getClassName())) |
| 49 | + return null; |
| 50 | + System.out.println("Inst: " + cr.getClassName()); |
| 51 | + ClassWriter cw = new ClassWriter(cr, ClassWriter.COMPUTE_MAXS | ClassWriter.COMPUTE_FRAMES); |
| 52 | + NonDeterministicReplayClassVisitor cv = new NonDeterministicReplayClassVisitor(Opcodes.ASM5, cw); |
| 53 | + cr.accept(cv, ClassReader.EXPAND_FRAMES); |
| 54 | + return cw.toByteArray(); |
| 55 | + } else { |
| 56 | + try { |
| 57 | + ClassReader cr = new ClassReader(classfileBuffer); |
| 58 | + if (isIgnoredClass(cr.getClassName())) |
| 59 | + return null; |
| 60 | + System.out.println("Inst: " + cr.getClassName()); |
| 61 | + ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS); |
| 62 | + NonDeterministicLoggingClassVisitor cv = new NonDeterministicLoggingClassVisitor(new SerialVersionUIDAdder(cw)); |
| 63 | + CallbackDuplicatingClassVisitor callbackDuplicator = new CallbackDuplicatingClassVisitor(cv); |
| 64 | + |
| 65 | + cr.accept(callbackDuplicator, ClassReader.EXPAND_FRAMES); |
| 66 | + return cw.toByteArray(); |
| 67 | + } catch (Throwable t) { |
| 68 | + t.printStackTrace(); |
| 69 | + return null; |
| 70 | + } |
| 71 | + } |
| 72 | + } |
| 73 | + }; |
| 74 | + inst.addTransformer(transformer); |
| 75 | + |
| 76 | + } |
| 77 | +} |
0 commit comments