Programs, and Systems, ECOOP 2016, Rome, Italy 10/63 Effort vs. expressiveness vs. control The AspectJ way High-level concepts, low effort, but limited expressiveness, control, and performance The ASM way High expressiveness, control, and performance, but bytecode-level concepts and high effort The DiSL way AspectJ-like (AOP-inspired) concepts ASM-like control, performance, and coverage for (MethodNode method : classNode.methods) { if ((method.access & (Opcodes.ACC_NATIVE | Opcodes.ACC_ABSTRACT)) != 0) { continue; } String methodName = method.name + "." + method.desc; InsnList instrumentation = new InsnList(); instrumentation.insert(new MethodInsnNode( Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V")); instrumentation.insert(new LdcInsnNode("On " + methodName + " method entry")); instrumentation.insert(new FieldInsnNode( Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;")); InsnList instructions = method.instructions; instructions.insert(instrumentation); instrumentation.clear(); instrumentation.insert(new MethodInsnNode( Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V")); instrumentation.insert(new LdcInsnNode("On " + methodName + " method exit")); instrumentation.insert(new FieldInsnNode( Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;")); for (AbstractInsnNode instruction : instructions.toArray()) { int opcode = instruction.getOpcode(); if (opcode >= Opcodes.IRETURN && opcode <= Opcodes.RETURN) { instructions.insert(instruction.getPrevious(), instrumentation); } } } for (MethodNode method : classNode.methods) { if ((method.access & (Opcodes.ACC_NATIVE | Opcodes.ACC_ABSTRACT)) != 0) { continue; } String methodName = method.name + "." + method.desc; InsnList instrumentation = new InsnList(); instrumentation.insert(new MethodInsnNode( Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V")); instrumentation.insert(new LdcInsnNode("On " + methodName + " method entry")); instrumentation.insert(new FieldInsnNode( Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;")); InsnList instructions = method.instructions; instructions.insert(instrumentation); instrumentation.clear(); instrumentation.insert(new MethodInsnNode( Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V")); instrumentation.insert(new LdcInsnNode("On " + methodName + " method exit")); instrumentation.insert(new FieldInsnNode( Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;")); for (AbstractInsnNode instruction : instructions.toArray()) { int opcode = instruction.getOpcode(); if (opcode >= Opcodes.IRETURN && opcode <= Opcodes.RETURN) { instructions.insert(instruction.getPrevious(), instrumentation); } } }