001// Licensed under the Apache License, Version 2.0 (the "License"); 002// you may not use this file except in compliance with the License. 003// You may obtain a copy of the License at 004// 005// http://www.apache.org/licenses/LICENSE-2.0 006// 007// Unless required by applicable law or agreed to in writing, software 008// distributed under the License is distributed on an "AS IS" BASIS, 009// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 010// See the License for the specific language governing permissions and 011// limitations under the License. 012 013package org.apache.tapestry5.internal.plastic; 014 015import org.apache.tapestry5.internal.plastic.asm.AnnotationVisitor; 016import org.apache.tapestry5.internal.plastic.asm.Opcodes; 017import org.apache.tapestry5.internal.plastic.asm.Type; 018import org.apache.tapestry5.internal.plastic.asm.tree.*; 019import org.apache.tapestry5.plastic.*; 020 021import java.io.IOException; 022import java.lang.annotation.Annotation; 023import java.lang.reflect.Constructor; 024import java.lang.reflect.Method; 025import java.lang.reflect.Modifier; 026import java.util.*; 027 028@SuppressWarnings("all") 029public class PlasticClassImpl extends Lockable implements PlasticClass, InternalPlasticClassTransformation, Opcodes 030{ 031 private static final String NOTHING_TO_VOID = "()V"; 032 033 static final String CONSTRUCTOR_NAME = "<init>"; 034 035 private static final String OBJECT_INT_TO_OBJECT = "(Ljava/lang/Object;I)Ljava/lang/Object;"; 036 037 private static final String OBJECT_INT_OBJECT_TO_VOID = "(Ljava/lang/Object;ILjava/lang/Object;)V"; 038 039 private static final String OBJECT_INT_OBJECT_ARRAY_TO_METHOD_INVOCATION_RESULT = String.format( 040 "(Ljava/lang/Object;I[Ljava/lang/Object;)%s", toDesc(Type.getInternalName(MethodInvocationResult.class))); 041 042 static final String ABSTRACT_METHOD_INVOCATION_INTERNAL_NAME = PlasticInternalUtils 043 .toInternalName(AbstractMethodInvocation.class.getName()); 044 045 private static final String HANDLE_SHIM_BASE_CLASS_INTERNAL_NAME = Type 046 .getInternalName(PlasticClassHandleShim.class); 047 048 static final String STATIC_CONTEXT_INTERNAL_NAME = Type.getInternalName(StaticContext.class); 049 050 private static final String INSTANCE_CONTEXT_INTERNAL_NAME = Type.getInternalName(InstanceContext.class); 051 052 private static final String INSTANCE_CONTEXT_DESC = toDesc(INSTANCE_CONTEXT_INTERNAL_NAME); 053 054 private static final String CONSTRUCTOR_DESC = String.format("(L%s;L%s;)V", STATIC_CONTEXT_INTERNAL_NAME, 055 INSTANCE_CONTEXT_INTERNAL_NAME); 056 057 static final Method STATIC_CONTEXT_GET_METHOD = toMethod(StaticContext.class, "get", int.class); 058 059 static final Method COMPUTED_VALUE_GET_METHOD = toMethod(ComputedValue.class, "get", InstanceContext.class); 060 061 private static final Method CONSTRUCTOR_CALLBACK_METHOD = toMethod(ConstructorCallback.class, "onConstruct", 062 Object.class, InstanceContext.class); 063 064 private static String toDesc(String internalName) 065 { 066 return "L" + internalName + ";"; 067 } 068 069 private static Method toMethod(Class declaringClass, String methodName, Class... parameterTypes) 070 { 071 return PlasticUtils.getMethod(declaringClass, methodName, parameterTypes); 072 } 073 074 static <T> T safeArrayDeref(T[] array, int index) 075 { 076 if (array == null) 077 return null; 078 079 return array[index]; 080 } 081 082 // Now past the inner classes; these are the instance variables of PlasticClassImpl proper: 083 084 final ClassNode classNode; 085 086 final PlasticClassPool pool; 087 088 private final boolean proxy; 089 090 final String className; 091 092 private final String superClassName; 093 094 private final AnnotationAccess annotationAccess; 095 096 // All the non-introduced (and non-constructor) methods, in sorted order 097 098 private final List<PlasticMethodImpl> methods; 099 100 private final Map<MethodDescription, PlasticMethod> description2method = new HashMap<>(); 101 102 final Set<String> methodNames = new HashSet<>(); 103 104 private final List<ConstructorCallback> constructorCallbacks = PlasticInternalUtils.newList(); 105 106 // All non-introduced instance fields 107 108 private final List<PlasticFieldImpl> fields; 109 110 /** 111 * Methods that require special attention inside {@link #createInstantiator()} because they 112 * have method advice. 113 */ 114 final Set<PlasticMethodImpl> advisedMethods = PlasticInternalUtils.newSet(); 115 116 final NameCache nameCache = new NameCache(); 117 118 // This is generated from fields, as necessary 119 List<PlasticField> unclaimedFields; 120 121 private final Set<String> fieldNames = PlasticInternalUtils.newSet(); 122 123 final StaticContext staticContext; 124 125 final InheritanceData parentInheritanceData, inheritanceData; 126 127 // MethodNodes in which field transformations should occur; this is most existing and 128 // introduced methods, outside of special access methods. 129 130 final Set<MethodNode> fieldTransformMethods = PlasticInternalUtils.newSet(); 131 132 // Tracks any methods that the Shim class uses to gain access to fields; used to ensure that 133 // such methods are not optimized away incorrectly. 134 final Set<MethodNode> shimInvokedMethods = PlasticInternalUtils.newSet(); 135 136 137 /** 138 * Tracks instrumentations of fields of this class, including private fields which are not published into the 139 * {@link PlasticClassPool}. 140 */ 141 private final FieldInstrumentations fieldInstrumentations; 142 143 /** 144 * This normal no-arguments constructor, or null. By the end of the transformation 145 * this will be converted into an ordinary method. 146 */ 147 private MethodNode originalConstructor; 148 149 private final MethodNode newConstructor; 150 151 final InstructionBuilder constructorBuilder; 152 153 private String instanceContextFieldName; 154 155 private Class<?> transformedClass; 156 157 // Indexes used to identify fields or methods in the shim 158 int nextFieldIndex = 0; 159 160 int nextMethodIndex = 0; 161 162 // Set of fields that need to contribute to the shim and gain access to it 163 164 final Set<PlasticFieldImpl> shimFields = PlasticInternalUtils.newSet(); 165 166 // Set of methods that need to contribute to the shim and gain access to it 167 168 final Set<PlasticMethodImpl> shimMethods = PlasticInternalUtils.newSet(); 169 170 final ClassNode implementationClassNode; 171 172 private ClassNode interfaceClassNode; 173 174 /** 175 * @param classNode 176 * @param implementationClassNode 177 * @param pool 178 * @param parentInheritanceData 179 * @param parentStaticContext 180 * @param proxy 181 */ 182 public PlasticClassImpl(ClassNode classNode, ClassNode implementationClassNode, PlasticClassPool pool, InheritanceData parentInheritanceData, 183 StaticContext parentStaticContext, boolean proxy) 184 { 185 this.classNode = classNode; 186 this.pool = pool; 187 this.proxy = proxy; 188 this.implementationClassNode = implementationClassNode; 189 190 staticContext = parentStaticContext.dupe(); 191 192 className = PlasticInternalUtils.toClassName(classNode.name); 193 superClassName = PlasticInternalUtils.toClassName(classNode.superName); 194 int lastIndexOfDot = className.lastIndexOf('.'); 195 196 String packageName = lastIndexOfDot > -1 ? className.substring(0, lastIndexOfDot) : ""; 197 198 fieldInstrumentations = new FieldInstrumentations(classNode.superName); 199 200 annotationAccess = new DelegatingAnnotationAccess(pool.createAnnotationAccess(classNode.visibleAnnotations), 201 pool.createAnnotationAccess(superClassName)); 202 203 this.parentInheritanceData = parentInheritanceData; 204 205 inheritanceData = parentInheritanceData.createChild(packageName); 206 207 for (String interfaceName : classNode.interfaces) 208 { 209 inheritanceData.addInterface(interfaceName); 210 } 211 212 methods = new ArrayList<>(classNode.methods.size()); 213 214 String invalidConstructorMessage = invalidConstructorMessage(); 215 216 for (MethodNode node : classNode.methods) 217 { 218 if (node.name.equals(CONSTRUCTOR_NAME)) 219 { 220 if (node.desc.equals(NOTHING_TO_VOID)) 221 { 222 originalConstructor = node; 223 fieldTransformMethods.add(node); 224 } else 225 { 226 node.instructions.clear(); 227 228 newBuilder(node).throwException(IllegalStateException.class, invalidConstructorMessage); 229 } 230 231 continue; 232 } 233 234 /* 235 * Static methods are not visible to the main API methods, but they must still be transformed, 236 * in case they directly access fields. In addition, track their names to avoid collisions. 237 */ 238 if (Modifier.isStatic(node.access)) 239 { 240 if (isInheritableMethod(node)) 241 { 242 inheritanceData.addMethod(node.name, node.desc, node.access == 0); 243 } 244 245 methodNames.add(node.name); 246 247 fieldTransformMethods.add(node); 248 249 continue; 250 } 251 252 if (!Modifier.isAbstract(node.access)) 253 { 254 fieldTransformMethods.add(node); 255 } 256 257 PlasticMethodImpl pmi = new PlasticMethodImpl(this, node); 258 259 methods.add(pmi); 260 description2method.put(pmi.getDescription(), pmi); 261 262 if (isInheritableMethod(node)) 263 { 264 inheritanceData.addMethod(node.name, node.desc, node.access == 0); 265 } 266 267 methodNames.add(node.name); 268 } 269 270 methodNames.addAll(parentInheritanceData.methodNames()); 271 272 Collections.sort(methods); 273 274 fields = new ArrayList<>(classNode.fields.size()); 275 276 for (FieldNode node : classNode.fields) 277 { 278 fieldNames.add(node.name); 279 280 // Ignore static fields. 281 282 if (Modifier.isStatic(node.access)) 283 continue; 284 285 // When we instrument the field such that it must be private, we'll get an exception. 286 287 fields.add(new PlasticFieldImpl(this, node)); 288 } 289 290 Collections.sort(fields); 291 292 // TODO: Make the output class's constructor protected, and create a shim class to instantiate it 293 // efficiently (without reflection). 294 newConstructor = new MethodNode(ACC_PUBLIC, CONSTRUCTOR_NAME, CONSTRUCTOR_DESC, null, null); 295 constructorBuilder = newBuilder(newConstructor); 296 297 // Start by calling the super-class no args constructor 298 299 if (parentInheritanceData.isTransformed()) 300 { 301 // If the parent is transformed, our first step is always to invoke its constructor. 302 303 constructorBuilder.loadThis().loadArgument(0).loadArgument(1); 304 constructorBuilder.invokeConstructor(superClassName, StaticContext.class.getName(), 305 InstanceContext.class.getName()); 306 } else 307 { 308 // Assumes the base class includes a visible constructor that takes no arguments. 309 // TODO: Do a proper check for this case and throw a meaningful exception 310 // if not present. 311 312 constructorBuilder.loadThis().invokeConstructor(superClassName); 313 } 314 315 // During the transformation, we'll be adding code to the constructor to pull values 316 // out of the static or instance context and assign them to fields. 317 318 // Later on, we'll add the RETURN opcode 319 } 320 321 private String invalidConstructorMessage() 322 { 323 return String.format("Class %s has been transformed and may not be directly instantiated.", className); 324 } 325 326 @Override 327 public <T extends Annotation> boolean hasAnnotation(Class<T> annotationType) 328 { 329 check(); 330 331 return annotationAccess.hasAnnotation(annotationType); 332 } 333 334 @Override 335 public <T extends Annotation> T getAnnotation(Class<T> annotationType) 336 { 337 check(); 338 339 return annotationAccess.getAnnotation(annotationType); 340 } 341 342 private static void addMethodAndParameterAnnotationsFromExistingClass(MethodNode methodNode, MethodNode implementationMethodNode) 343 { 344 // visits the method attributes 345 int i, j, n; 346 if (implementationMethodNode.annotationDefault != null) 347 { 348 AnnotationVisitor av = methodNode.visitAnnotationDefault(); 349 TapestryAnnotationNode.accept(av, null, implementationMethodNode.annotationDefault); 350 if (av != null) 351 { 352 av.visitEnd(); 353 } 354 } 355 n = implementationMethodNode.visibleAnnotations == null ? 0 : implementationMethodNode.visibleAnnotations.size(); 356 for (i = 0; i < n; ++i) 357 { 358 AnnotationNode an = implementationMethodNode.visibleAnnotations.get(i); 359 an.accept(methodNode.visitAnnotation(an.desc, true)); 360 } 361 n = implementationMethodNode.invisibleAnnotations == null ? 0 : implementationMethodNode.invisibleAnnotations.size(); 362 for (i = 0; i < n; ++i) 363 { 364 AnnotationNode an = implementationMethodNode.invisibleAnnotations.get(i); 365 an.accept(methodNode.visitAnnotation(an.desc, false)); 366 } 367 n = implementationMethodNode.visibleParameterAnnotations == null 368 ? 0 369 : implementationMethodNode.visibleParameterAnnotations.length; 370 for (i = 0; i < n; ++i) 371 { 372 List<?> l = implementationMethodNode.visibleParameterAnnotations[i]; 373 if (l == null) 374 { 375 continue; 376 } 377 for (j = 0; j < l.size(); ++j) 378 { 379 AnnotationNode an = (AnnotationNode) l.get(j); 380 an.accept(methodNode.visitParameterAnnotation(i, an.desc, true)); 381 } 382 } 383 n = implementationMethodNode.invisibleParameterAnnotations == null 384 ? 0 385 : implementationMethodNode.invisibleParameterAnnotations.length; 386 for (i = 0; i < n; ++i) 387 { 388 List<?> l = implementationMethodNode.invisibleParameterAnnotations[i]; 389 if (l == null) 390 { 391 continue; 392 } 393 for (j = 0; j < l.size(); ++j) 394 { 395 AnnotationNode an = (AnnotationNode) l.get(j); 396 an.accept(methodNode.visitParameterAnnotation(i, an.desc, false)); 397 } 398 } 399 400 methodNode.visitEnd(); 401 402 } 403 404 private static void removeDuplicatedAnnotations(MethodNode node) 405 { 406 407 removeDuplicatedAnnotations(node.visibleAnnotations); 408 removeDuplicatedAnnotations(node.invisibleAnnotations); 409 410 if (node.visibleParameterAnnotations != null) 411 { 412 for (List<AnnotationNode> list : node.visibleParameterAnnotations) 413 { 414 removeDuplicatedAnnotations(list); 415 } 416 } 417 418 if (node.invisibleParameterAnnotations != null) 419 { 420 for (List<AnnotationNode> list : node.invisibleParameterAnnotations) 421 { 422 removeDuplicatedAnnotations(list); 423 } 424 } 425 426 } 427 428 private static void removeDuplicatedAnnotations(ClassNode node) 429 { 430 removeDuplicatedAnnotations(node.visibleAnnotations, true); 431 removeDuplicatedAnnotations(node.invisibleAnnotations, true); 432 } 433 434 private static void removeDuplicatedAnnotations(List<AnnotationNode> list) { 435 removeDuplicatedAnnotations(list, false); 436 } 437 438 private static void removeDuplicatedAnnotations(List<AnnotationNode> list, boolean reverse) { 439 440 if (list != null) 441 { 442 443 final Set<String> annotations = new HashSet<>(); 444 final List<AnnotationNode> toBeRemoved = new ArrayList<>(); 445 final List<AnnotationNode> toBeIterated; 446 447 if (reverse) 448 { 449 toBeIterated = new ArrayList<>(list); 450 Collections.reverse(toBeIterated); 451 } 452 else { 453 toBeIterated = list; 454 } 455 456 for (AnnotationNode annotationNode : toBeIterated) 457 { 458 if (annotations.contains(annotationNode.desc)) 459 { 460 toBeRemoved.add(annotationNode); 461 } 462 else 463 { 464 annotations.add(annotationNode.desc); 465 } 466 } 467 468 for (AnnotationNode annotationNode : toBeRemoved) 469 { 470 list.remove(annotationNode); 471 } 472 473 } 474 475 } 476 477 private static String getParametersDesc(MethodNode methodNode) { 478 return methodNode.desc.substring(methodNode.desc.indexOf('(') + 1, methodNode.desc.lastIndexOf(')')); 479 } 480 481 private static MethodNode findExactMatchMethod(MethodNode methodNode, ClassNode source) { 482 483 MethodNode found = null; 484 485 final String methodDescription = getParametersDesc(methodNode); 486 487 for (MethodNode implementationMethodNode : source.methods) 488 { 489 490 final String implementationMethodDescription = getParametersDesc(implementationMethodNode); 491 if (methodNode.name.equals(implementationMethodNode.name) && 492 // We don't want synthetic methods. 493 ((implementationMethodNode.access & Opcodes.ACC_SYNTHETIC) == 0) 494 && (methodDescription.equals(implementationMethodDescription))) 495 { 496 found = implementationMethodNode; 497 break; 498 } 499 } 500 501 return found; 502 503 } 504 505 private static List<Class> getJavaParameterTypes(MethodNode methodNode) { 506 final ClassLoader classLoader = PlasticInternalUtils.class.getClassLoader(); 507 Type[] parameterTypes = Type.getArgumentTypes(methodNode.desc); 508 List<Class> list = new ArrayList<>(); 509 for (Type type : parameterTypes) 510 { 511 try 512 { 513 list.add(PlasticInternalUtils.toClass(classLoader, type.getClassName())); 514 } 515 catch (ClassNotFoundException e) 516 { 517 throw new RuntimeException(e); // shouldn't happen anyway 518 } 519 } 520 return list; 521 } 522 523 /** 524 * Returns the first method which matches the given methodNode. 525 * FIXME: this may not find the correct method if the correct one is declared after 526 * another in which all parameters are supertypes of the parameters of methodNode. 527 * To solve this, we would need to dig way deeper than we have time for this. 528 * @param methodNode 529 * @param classNode 530 * @return 531 */ 532 private static MethodNode findGenericMethod(MethodNode methodNode, ClassNode classNode) 533 { 534 535 MethodNode found = null; 536 537 List<Class> parameterTypes = getJavaParameterTypes(methodNode); 538 539 for (MethodNode implementationMethodNode : classNode.methods) 540 { 541 542 if (methodNode.name.equals(implementationMethodNode.name)) 543 { 544 545 final List<Class> implementationParameterTypes = getJavaParameterTypes(implementationMethodNode); 546 547 if (parameterTypes.size() == implementationParameterTypes.size()) 548 { 549 550 boolean matches = true; 551 for (int i = 0; i < parameterTypes.size(); i++) 552 { 553 final Class implementationParameterType = implementationParameterTypes.get(i); 554 final Class parameterType = parameterTypes.get(i); 555 if (!parameterType.isAssignableFrom(implementationParameterType)) { 556 matches = false; 557 break; 558 } 559 560 } 561 562 if (matches && !isBridge(implementationMethodNode)) 563 { 564 found = implementationMethodNode; 565 break; 566 } 567 568 } 569 570 } 571 572 } 573 574 return found; 575 576 } 577 578 private static void addMethodAndParameterAnnotationsFromExistingClass(MethodNode methodNode, ClassNode source) 579 { 580 if (source != null) 581 { 582 583 MethodNode candidate = findExactMatchMethod(methodNode, source); 584 585 final String parametersDesc = getParametersDesc(methodNode); 586 587 // candidate will be null when the method has generic parameters 588 if (candidate == null && parametersDesc.trim().length() > 0) 589 { 590 candidate = findGenericMethod(methodNode, source); 591 } 592 593 if (candidate != null) 594 { 595 addMethodAndParameterAnnotationsFromExistingClass(methodNode, candidate); 596 } 597 598 } 599 600 } 601 602 /** 603 * Tells whether a given method is a bridge one or not. 604 * Notice the flag for bridge method is the same as volatile field. Java 6 doesn't have 605 * Modifiers.isBridge(), so we use a workaround. 606 */ 607 private static boolean isBridge(MethodNode methodNode) 608 { 609 return Modifier.isVolatile(methodNode.access); 610 } 611 612 @Override 613 public PlasticClass proxyInterface(Class interfaceType, PlasticField field) 614 { 615 check(); 616 617 assert field != null; 618 619 introduceInterface(interfaceType); 620 621 for (Method m : getUniqueMethods(interfaceType)) 622 { 623 final MethodDescription description = new MethodDescription(m); 624 if(Modifier.isStatic(description.modifiers)) 625 { 626 continue; 627 } 628 introduceMethod(description).delegateTo(field); 629 } 630 631 return this; 632 } 633 634 @Override 635 public ClassInstantiator createInstantiator() 636 { 637 lock(); 638 639 addClassAnnotations(implementationClassNode); 640 removeDuplicatedAnnotations(classNode); 641 642 createShimIfNeeded(); 643 644 interceptFieldAccess(); 645 646 rewriteAdvisedMethods(); 647 648 completeConstructor(); 649 650 transformedClass = pool.realizeTransformedClass(classNode, inheritanceData, staticContext); 651 652 return createInstantiatorFromClass(transformedClass); 653 } 654 655 private void addClassAnnotations(ClassNode otherClassNode) 656 { 657 // Copy annotations from implementation if available. 658 // Code adapted from ClassNode.accept(), as we just want to copy 659 // the annotations and nothing more. 660 if (otherClassNode != null) 661 { 662 663 int i, n; 664 n = otherClassNode.visibleAnnotations == null ? 0 : otherClassNode.visibleAnnotations.size(); 665 for (i = 0; i < n; ++i) 666 { 667 AnnotationNode an = otherClassNode.visibleAnnotations.get(i); 668 an.accept(classNode.visitAnnotation(an.desc, true)); 669 } 670 n = otherClassNode.invisibleAnnotations == null ? 0 : otherClassNode.invisibleAnnotations.size(); 671 for (i = 0; i < n; ++i) 672 { 673 AnnotationNode an = otherClassNode.invisibleAnnotations.get(i); 674 an.accept(classNode.visitAnnotation(an.desc, false)); 675 } 676 677 } 678 } 679 680 private ClassInstantiator createInstantiatorFromClass(Class clazz) 681 { 682 try 683 { 684 Constructor ctor = clazz.getConstructor(StaticContext.class, InstanceContext.class); 685 686 return new ClassInstantiatorImpl(clazz, ctor, staticContext); 687 } catch (Exception ex) 688 { 689 throw new RuntimeException(String.format("Unable to create ClassInstantiator for class %s: %s", 690 clazz.getName(), PlasticInternalUtils.toMessage(ex)), ex); 691 } 692 } 693 694 private void completeConstructor() 695 { 696 if (originalConstructor != null) 697 { 698 convertOriginalConstructorToMethod(); 699 } 700 701 invokeCallbacks(); 702 703 constructorBuilder.returnResult(); 704 705 classNode.methods.add(newConstructor); 706 } 707 708 private void invokeCallbacks() 709 { 710 for (ConstructorCallback callback : constructorCallbacks) 711 { 712 invokeCallback(callback); 713 } 714 } 715 716 private void invokeCallback(ConstructorCallback callback) 717 { 718 int index = staticContext.store(callback); 719 720 // First, load the callback 721 722 constructorBuilder.loadArgument(0).loadConstant(index).invoke(STATIC_CONTEXT_GET_METHOD).castOrUnbox(ConstructorCallback.class.getName()); 723 724 // Load this and the InstanceContext 725 constructorBuilder.loadThis().loadArgument(1); 726 727 constructorBuilder.invoke(CONSTRUCTOR_CALLBACK_METHOD); 728 } 729 730 731 /** 732 * Convert the original constructor into a private method invoked from the 733 * generated constructor. 734 */ 735 private void convertOriginalConstructorToMethod() 736 { 737 String initializerName = makeUnique(methodNames, "initializeInstance"); 738 739 int originalAccess = originalConstructor.access; 740 741 originalConstructor.access = ACC_PRIVATE; 742 originalConstructor.name = initializerName; 743 744 stripOutSuperConstructorCall(originalConstructor); 745 746 constructorBuilder.loadThis().invokeVirtual(className, "void", initializerName); 747 748 // And replace it with a constructor that throws an exception 749 750 MethodNode replacementConstructor = new MethodNode(originalAccess, CONSTRUCTOR_NAME, NOTHING_TO_VOID, null, 751 null); 752 753 newBuilder(replacementConstructor).throwException(IllegalStateException.class, invalidConstructorMessage()); 754 755 classNode.methods.add(replacementConstructor); 756 } 757 758 private void stripOutSuperConstructorCall(MethodNode cons) 759 { 760 InsnList ins = cons.instructions; 761 762 ListIterator li = ins.iterator(); 763 764 // Look for the ALOAD 0 (i.e., push this on the stack) 765 while (li.hasNext()) 766 { 767 AbstractInsnNode node = (AbstractInsnNode) li.next(); 768 769 if (node.getOpcode() == ALOAD) 770 { 771 VarInsnNode varNode = (VarInsnNode) node; 772 773 assert varNode.var == 0; 774 775 // Remove the ALOAD 776 li.remove(); 777 break; 778 } 779 } 780 781 // Look for the call to the super-class, an INVOKESPECIAL 782 while (li.hasNext()) 783 { 784 AbstractInsnNode node = (AbstractInsnNode) li.next(); 785 786 if (node.getOpcode() == INVOKESPECIAL) 787 { 788 MethodInsnNode mnode = (MethodInsnNode) node; 789 790 assert mnode.owner.equals(classNode.superName); 791 assert mnode.name.equals(CONSTRUCTOR_NAME); 792 assert mnode.desc.equals(cons.desc); 793 794 li.remove(); 795 return; 796 } 797 } 798 799 throw new AssertionError("Could not convert constructor to simple method."); 800 } 801 802 @Override 803 public <T extends Annotation> List<PlasticField> getFieldsWithAnnotation(Class<T> annotationType) 804 { 805 check(); 806 807 List<PlasticField> result = getAllFields(); 808 809 Iterator<PlasticField> iterator = result.iterator(); 810 811 while (iterator.hasNext()) 812 { 813 PlasticField plasticField = iterator.next(); 814 815 if (!plasticField.hasAnnotation(annotationType)) 816 iterator.remove(); 817 } 818 819 return result; 820 } 821 822 @Override 823 public List<PlasticField> getAllFields() 824 { 825 check(); 826 827 return new ArrayList<>(fields); 828 } 829 830 @Override 831 public List<PlasticField> getUnclaimedFields() 832 { 833 check(); 834 835 // Initially null, and set back to null by PlasticField.claim(). 836 837 if (unclaimedFields == null) 838 { 839 unclaimedFields = new ArrayList<>(fields.size()); 840 841 for (PlasticField f : fields) 842 { 843 if (!f.isClaimed()) 844 unclaimedFields.add(f); 845 } 846 } 847 848 return unclaimedFields; 849 } 850 851 @Override 852 public PlasticMethod introducePrivateMethod(String typeName, String suggestedName, String[] argumentTypes, 853 String[] exceptionTypes) 854 { 855 check(); 856 857 assert PlasticInternalUtils.isNonBlank(typeName); 858 assert PlasticInternalUtils.isNonBlank(suggestedName); 859 860 String name = makeUnique(methodNames, suggestedName); 861 862 MethodDescription description = new MethodDescription(Modifier.PRIVATE, typeName, name, argumentTypes, null, 863 exceptionTypes); 864 865 return introduceMethod(description); 866 } 867 868 @Override 869 public PlasticField introduceField(String className, String suggestedName) 870 { 871 check(); 872 873 assert PlasticInternalUtils.isNonBlank(className); 874 assert PlasticInternalUtils.isNonBlank(suggestedName); 875 876 String name = makeUnique(fieldNames, suggestedName); 877 878 // No signature and no initial value 879 880 FieldNode fieldNode = new FieldNode(ACC_PRIVATE, name, PlasticInternalUtils.toDescriptor(className), null, null); 881 882 classNode.fields.add(fieldNode); 883 884 fieldNames.add(name); 885 886 PlasticFieldImpl newField = new PlasticFieldImpl(this, fieldNode); 887 888 return newField; 889 } 890 891 @Override 892 public PlasticField introduceField(Class fieldType, String suggestedName) 893 { 894 assert fieldType != null; 895 896 return introduceField(nameCache.toTypeName(fieldType), suggestedName); 897 } 898 899 String makeUnique(Set<String> values, String input) 900 { 901 return values.contains(input) ? input + "$" + PlasticUtils.nextUID() : input; 902 } 903 904 @Override 905 public <T extends Annotation> List<PlasticMethod> getMethodsWithAnnotation(Class<T> annotationType) 906 { 907 check(); 908 909 List<PlasticMethod> result = getMethods(); 910 911 result.removeIf(method -> !method.hasAnnotation(annotationType)); 912 913 return result; 914 } 915 916 @Override 917 public List<PlasticMethod> getMethods() 918 { 919 check(); 920 921 return new ArrayList<>(methods); 922 } 923 924 @Override 925 public PlasticMethod introduceMethod(MethodDescription description) 926 { 927 check(); 928 929 if (Modifier.isAbstract(description.modifiers)) 930 { 931 description = description.withModifiers(description.modifiers & ~ACC_ABSTRACT); 932 } 933 934 PlasticMethod result = description2method.get(description); 935 936 if (result == null) 937 { 938 result = createNewMethod(description); 939 940 description2method.put(description, result); 941 } 942 943 methodNames.add(description.methodName); 944 945 // Note that is it not necessary to add the new MethodNode to 946 // fieldTransformMethods (the default implementations provided by introduceMethod() do not 947 // ever access instance fields) ... unless the caller invokes changeImplementation(). 948 949 return result; 950 } 951 952 @Override 953 public PlasticMethod introduceMethod(MethodDescription description, InstructionBuilderCallback callback) 954 { 955 check(); 956 957 // TODO: optimize this so that a default implementation is not created. 958 959 return introduceMethod(description).changeImplementation(callback); 960 } 961 962 @Override 963 public PlasticMethod introduceMethod(Method method) 964 { 965 check(); 966 return introduceMethod(new MethodDescription(method)); 967 } 968 969 void addMethod(MethodNode methodNode) 970 { 971 classNode.methods.add(methodNode); 972 973 methodNames.add(methodNode.name); 974 975 if (isInheritableMethod(methodNode)) 976 { 977 inheritanceData.addMethod(methodNode.name, methodNode.desc, methodNode.access == 0); 978 } 979 } 980 981 private PlasticMethod createNewMethod(MethodDescription description) 982 { 983 if (Modifier.isStatic(description.modifiers)) 984 throw new IllegalArgumentException(String.format( 985 "Unable to introduce method '%s' into class %s: introduced methods may not be static.", 986 description, className)); 987 988 String desc = nameCache.toDesc(description); 989 990 String[] exceptions = new String[description.checkedExceptionTypes.length]; 991 for (int i = 0; i < exceptions.length; i++) 992 { 993 exceptions[i] = PlasticInternalUtils.toInternalName(description.checkedExceptionTypes[i]); 994 } 995 996 MethodNode methodNode = new MethodNode(description.modifiers, description.methodName, desc, 997 description.genericSignature, exceptions); 998 boolean isOverride = inheritanceData.isImplemented(methodNode.name, desc); 999 1000 if (!isOverride) 1001 { 1002 addMethodAndParameterAnnotationsFromExistingClass(methodNode, implementationClassNode); 1003 addMethodAndParameterAnnotationsFromExistingClass(methodNode, interfaceClassNode); 1004 removeDuplicatedAnnotations(methodNode); 1005 } 1006 1007 if (isOverride) 1008 createOverrideOfBaseClassImpl(description, methodNode); 1009 else 1010 createNewMethodImpl(description, methodNode); 1011 1012 addMethod(methodNode); 1013 1014 return new PlasticMethodImpl(this, methodNode); 1015 } 1016 1017 private void createNewMethodImpl(MethodDescription methodDescription, MethodNode methodNode) 1018 { 1019 newBuilder(methodDescription, methodNode).returnDefaultValue(); 1020 } 1021 1022 private void createOverrideOfBaseClassImpl(MethodDescription methodDescription, MethodNode methodNode) 1023 { 1024 InstructionBuilder builder = newBuilder(methodDescription, methodNode); 1025 1026 builder.loadThis(); 1027 builder.loadArguments(); 1028 builder.invokeSpecial(superClassName, methodDescription); 1029 builder.returnResult(); 1030 } 1031 1032 /** 1033 * Iterates over all non-introduced methods, including the original constructor. For each 1034 * method, the bytecode is scanned for field reads and writes. When a match is found against an intercepted field, 1035 * the operation is replaced with a method invocation. This is invoked only after the {@link PlasticClassHandleShim} 1036 * for the class has been created, as the shim may create methods that contain references to fields that may be 1037 * subject to field access interception. 1038 */ 1039 private void interceptFieldAccess() 1040 { 1041 for (MethodNode node : fieldTransformMethods) 1042 { 1043 // Intercept field access inside the method, tracking which access methods 1044 // are actually used by removing them from accessMethods 1045 1046 interceptFieldAccess(node); 1047 } 1048 } 1049 1050 /** 1051 * Determines if any fields or methods have provided FieldHandles or MethodHandles; if so 1052 * a shim class must be created to facilitate read/write access to fields, or invocation of methods. 1053 */ 1054 private void createShimIfNeeded() 1055 { 1056 if (shimFields.isEmpty() && shimMethods.isEmpty()) 1057 return; 1058 1059 PlasticClassHandleShim shim = createShimInstance(); 1060 1061 installShim(shim); 1062 } 1063 1064 public void installShim(PlasticClassHandleShim shim) 1065 { 1066 for (PlasticFieldImpl f : shimFields) 1067 { 1068 f.installShim(shim); 1069 } 1070 1071 for (PlasticMethodImpl m : shimMethods) 1072 { 1073 m.installShim(shim); 1074 } 1075 } 1076 1077 public PlasticClassHandleShim createShimInstance() 1078 { 1079 String shimClassName = String.format("%s$Shim_%s", classNode.name, PlasticUtils.nextUID()); 1080 1081 ClassNode shimClassNode = new ClassNode(); 1082 1083 shimClassNode.visit(PlasticConstants.DEFAULT_VERSION_OPCODE, ACC_PUBLIC | ACC_FINAL, shimClassName, null, HANDLE_SHIM_BASE_CLASS_INTERNAL_NAME, 1084 null); 1085 1086 implementConstructor(shimClassNode); 1087 1088 if (!shimFields.isEmpty()) 1089 { 1090 implementShimGet(shimClassNode); 1091 implementShimSet(shimClassNode); 1092 } 1093 1094 if (!shimMethods.isEmpty()) 1095 { 1096 implementShimInvoke(shimClassNode); 1097 } 1098 1099 return instantiateShim(shimClassNode); 1100 } 1101 1102 private void implementConstructor(ClassNode shimClassNode) 1103 { 1104 MethodNode mn = new MethodNode(ACC_PUBLIC, CONSTRUCTOR_NAME, NOTHING_TO_VOID, null, null); 1105 1106 InstructionBuilder builder = newBuilder(mn); 1107 1108 builder.loadThis().invokeConstructor(PlasticClassHandleShim.class).returnResult(); 1109 1110 shimClassNode.methods.add(mn); 1111 1112 } 1113 1114 private PlasticClassHandleShim instantiateShim(ClassNode shimClassNode) 1115 { 1116 try 1117 { 1118 Class shimClass = pool.realize(className, ClassType.SUPPORT, shimClassNode); 1119 1120 return (PlasticClassHandleShim) shimClass.newInstance(); 1121 } catch (Exception ex) 1122 { 1123 throw new RuntimeException( 1124 String.format("Unable to instantiate shim class %s for plastic class %s: %s", 1125 PlasticInternalUtils.toClassName(shimClassNode.name), className, 1126 PlasticInternalUtils.toMessage(ex)), ex); 1127 } 1128 } 1129 1130 private void implementShimGet(ClassNode shimClassNode) 1131 { 1132 MethodNode mn = new MethodNode(ACC_PUBLIC, "get", OBJECT_INT_TO_OBJECT, null, null); 1133 1134 InstructionBuilder builder = newBuilder(mn); 1135 1136 // Arg 0 is the target instance 1137 // Arg 1 is the index 1138 1139 builder.loadArgument(0).checkcast(className); 1140 builder.loadArgument(1); 1141 1142 builder.startSwitch(0, nextFieldIndex - 1, new SwitchCallback() 1143 { 1144 @Override 1145 public void doSwitch(SwitchBlock block) 1146 { 1147 for (PlasticFieldImpl f : shimFields) 1148 { 1149 f.extendShimGet(block); 1150 } 1151 } 1152 }); 1153 1154 shimClassNode.methods.add(mn); 1155 } 1156 1157 private void implementShimSet(ClassNode shimClassNode) 1158 { 1159 MethodNode mn = new MethodNode(ACC_PUBLIC, "set", OBJECT_INT_OBJECT_TO_VOID, null, null); 1160 1161 InstructionBuilder builder = newBuilder(mn); 1162 1163 // Arg 0 is the target instance 1164 // Arg 1 is the index 1165 // Arg 2 is the new value 1166 1167 builder.loadArgument(0).checkcast(className); 1168 builder.loadArgument(2); 1169 1170 builder.loadArgument(1); 1171 1172 builder.startSwitch(0, nextFieldIndex - 1, new SwitchCallback() 1173 { 1174 @Override 1175 public void doSwitch(SwitchBlock block) 1176 { 1177 for (PlasticFieldImpl f : shimFields) 1178 { 1179 f.extendShimSet(block); 1180 } 1181 } 1182 }); 1183 1184 builder.returnResult(); 1185 1186 shimClassNode.methods.add(mn); 1187 } 1188 1189 private void implementShimInvoke(ClassNode shimClassNode) 1190 { 1191 MethodNode mn = new MethodNode(ACC_PUBLIC, "invoke", OBJECT_INT_OBJECT_ARRAY_TO_METHOD_INVOCATION_RESULT, null, 1192 null); 1193 1194 InstructionBuilder builder = newBuilder(mn); 1195 1196 // Arg 0 is the target instance 1197 // Arg 1 is the index 1198 // Arg 2 is the object array of parameters 1199 1200 builder.loadArgument(0).checkcast(className); 1201 1202 builder.loadArgument(1); 1203 1204 builder.startSwitch(0, nextMethodIndex - 1, new SwitchCallback() 1205 { 1206 @Override 1207 public void doSwitch(SwitchBlock block) 1208 { 1209 for (PlasticMethodImpl m : shimMethods) 1210 { 1211 m.extendShimInvoke(block); 1212 } 1213 } 1214 }); 1215 1216 shimClassNode.methods.add(mn); 1217 } 1218 1219 private void rewriteAdvisedMethods() 1220 { 1221 for (PlasticMethodImpl method : advisedMethods) 1222 { 1223 method.rewriteMethodForAdvice(); 1224 } 1225 } 1226 1227 private void interceptFieldAccess(MethodNode methodNode) 1228 { 1229 InsnList insns = methodNode.instructions; 1230 1231 ListIterator<AbstractInsnNode> it = insns.iterator(); 1232 1233 while (it.hasNext()) 1234 { 1235 AbstractInsnNode node = it.next(); 1236 1237 int opcode = node.getOpcode(); 1238 1239 if (opcode != GETFIELD && opcode != PUTFIELD) 1240 { 1241 continue; 1242 } 1243 1244 FieldInsnNode fnode = (FieldInsnNode) node; 1245 1246 FieldInstrumentation instrumentation = findFieldNodeInstrumentation(fnode, opcode == GETFIELD); 1247 1248 if (instrumentation == null) 1249 { 1250 continue; 1251 } 1252 1253 // Replace the field access node with the appropriate method invocation. 1254 1255 insns.insertBefore(fnode, new MethodInsnNode(INVOKEVIRTUAL, fnode.owner, instrumentation.methodName, instrumentation.methodDescription, false)); 1256 1257 it.remove(); 1258 } 1259 } 1260 1261 private FieldInstrumentation findFieldNodeInstrumentation(FieldInsnNode node, boolean forRead) 1262 { 1263 // First look in the local fieldInstrumentations, which contains private field instrumentations 1264 // (as well as non-private ones). 1265 1266 String searchStart = node.owner; 1267 1268 if (searchStart.equals(classNode.name)) 1269 { 1270 FieldInstrumentation result = fieldInstrumentations.get(node.name, forRead); 1271 1272 if (result != null) 1273 { 1274 return result; 1275 } 1276 1277 // Slight optimization: start the search in the super-classes' fields, since we've already 1278 // checked this classes fields. 1279 1280 searchStart = classNode.superName; 1281 } 1282 1283 return pool.getFieldInstrumentation(searchStart, node.name, forRead); 1284 } 1285 1286 String getInstanceContextFieldName() 1287 { 1288 if (instanceContextFieldName == null) 1289 { 1290 instanceContextFieldName = makeUnique(fieldNames, "instanceContext"); 1291 1292 // TODO: We could use a protected field and only initialize 1293 // it once, in the first base class where it is needed, though that raises the possibilities 1294 // of name conflicts (a subclass might introduce a field with a conflicting name). 1295 1296 FieldNode node = new FieldNode(ACC_PRIVATE | ACC_FINAL, instanceContextFieldName, INSTANCE_CONTEXT_DESC, 1297 null, null); 1298 1299 classNode.fields.add(node); 1300 1301 // Extend the constructor to store the context in a field. 1302 1303 constructorBuilder.loadThis().loadArgument(1) 1304 .putField(className, instanceContextFieldName, InstanceContext.class); 1305 } 1306 1307 return instanceContextFieldName; 1308 } 1309 1310 /** 1311 * Creates a new private final field and initializes its value (using the StaticContext). 1312 */ 1313 String createAndInitializeFieldFromStaticContext(String suggestedFieldName, String fieldType, 1314 Object injectedFieldValue) 1315 { 1316 String name = makeUnique(fieldNames, suggestedFieldName); 1317 1318 FieldNode field = new FieldNode(ACC_PRIVATE | ACC_FINAL, name, nameCache.toDesc(fieldType), null, null); 1319 1320 classNode.fields.add(field); 1321 1322 initializeFieldFromStaticContext(name, fieldType, injectedFieldValue); 1323 1324 return name; 1325 } 1326 1327 /** 1328 * Initializes a field from the static context. The injected value is added to the static 1329 * context and the class constructor updated to assign the value from the context (which includes casting and 1330 * possibly unboxing). 1331 */ 1332 void initializeFieldFromStaticContext(String fieldName, String fieldType, Object injectedFieldValue) 1333 { 1334 int index = staticContext.store(injectedFieldValue); 1335 1336 // Although it feels nicer to do the loadThis() later and then swap(), that breaks 1337 // on primitive longs and doubles, so its just easier to do the loadThis() first 1338 // so its at the right place on the stack for the putField(). 1339 1340 constructorBuilder.loadThis(); 1341 1342 constructorBuilder.loadArgument(0).loadConstant(index); 1343 constructorBuilder.invoke(STATIC_CONTEXT_GET_METHOD); 1344 constructorBuilder.castOrUnbox(fieldType); 1345 1346 constructorBuilder.putField(className, fieldName, fieldType); 1347 } 1348 1349 void pushInstanceContextFieldOntoStack(InstructionBuilder builder) 1350 { 1351 builder.loadThis().getField(className, getInstanceContextFieldName(), InstanceContext.class); 1352 } 1353 1354 @Override 1355 public PlasticClass getPlasticClass() 1356 { 1357 return this; 1358 } 1359 1360 @Override 1361 public Class<?> getTransformedClass() 1362 { 1363 if (transformedClass == null) 1364 throw new IllegalStateException(String.format( 1365 "Transformed class %s is not yet available because the transformation is not yet complete.", 1366 className)); 1367 1368 return transformedClass; 1369 } 1370 1371 private boolean isInheritableMethod(MethodNode node) 1372 { 1373 return !Modifier.isPrivate(node.access); 1374 } 1375 1376 @Override 1377 public String getClassName() 1378 { 1379 return className; 1380 } 1381 1382 InstructionBuilderImpl newBuilder(MethodNode mn) 1383 { 1384 return newBuilder(PlasticInternalUtils.toMethodDescription(mn), mn); 1385 } 1386 1387 InstructionBuilderImpl newBuilder(MethodDescription description, MethodNode mn) 1388 { 1389 return new InstructionBuilderImpl(description, mn, nameCache); 1390 } 1391 1392 public Set<PlasticMethod> introduceInterface(Class interfaceType) 1393 { 1394 return introduceInterface(interfaceType, null); 1395 } 1396 1397 private Set<PlasticMethod> introduceInterface(Class interfaceType, PlasticMethod method) 1398 { 1399 check(); 1400 1401 assert interfaceType != null; 1402 1403 if (!interfaceType.isInterface()) 1404 throw new IllegalArgumentException(String.format( 1405 "Class %s is not an interface; only interfaces may be introduced.", interfaceType.getName())); 1406 1407 String interfaceName = nameCache.toInternalName(interfaceType); 1408 1409 try 1410 { 1411 interfaceClassNode = PlasticClassPool.readClassNode(interfaceType.getName(), getClass().getClassLoader()); 1412 } catch (IOException e) 1413 { 1414 throw new RuntimeException(e); 1415 } 1416 1417 if (!inheritanceData.isInterfaceImplemented(interfaceName)) 1418 { 1419 classNode.interfaces.add(interfaceName); 1420 inheritanceData.addInterface(interfaceName); 1421 } 1422 1423 addClassAnnotations(interfaceClassNode); 1424 1425 Set<PlasticMethod> introducedMethods = new HashSet<PlasticMethod>(); 1426 Set<Method> alreadyIntroducedMethods = new HashSet<>(); 1427 1428 Method[] sortedMethods = interfaceType.getMethods(); 1429 Arrays.sort(sortedMethods, METHOD_COMPARATOR); 1430 for (Method m : sortedMethods) 1431 { 1432 MethodDescription description = new MethodDescription(m); 1433 1434 if (!isMethodImplemented(description) && 1435 !(m.isDefault() && m.isBridge()) && 1436 !Modifier.isStatic(description.modifiers) && 1437 !contains(alreadyIntroducedMethods, m)) 1438 { 1439 PlasticMethod introducedMethod = introduceMethod(m); 1440 introducedMethods.add(introducedMethod); 1441 if (method != null) { 1442 introducedMethod.delegateTo(method); 1443 } 1444 alreadyIntroducedMethods.add(m); 1445 } 1446 } 1447 1448 interfaceClassNode = null; 1449 1450 return introducedMethods; 1451 } 1452 1453 @Override 1454 public PlasticClass proxyInterface(Class interfaceType, PlasticMethod method) 1455 { 1456 check(); 1457 assert method != null; 1458 1459 introduceInterface(interfaceType, method); 1460 1461 return this; 1462 } 1463 1464 private boolean contains(Set<Method> alreadyIntroducedMethods, Method m) { 1465 boolean contains = false; 1466 for (Method method : alreadyIntroducedMethods) 1467 { 1468 if (METHOD_COMPARATOR.compare(method, m) == 0) 1469 { 1470 contains = true; 1471 break; 1472 } 1473 } 1474 return false; 1475 } 1476 1477 private Map<MethodSignature, MethodDescription> createMethodSignatureMap(Class interfaceType) { 1478 // TAP-2582: preprocessing the method list so we don't add duplicated 1479 // methods, something that happens when an interface has superinterfaces 1480 // and they define the same method signature. 1481 // In addition, we collect all the thrown checked exceptions, just in case. 1482 Map<MethodSignature, MethodDescription> map = new HashMap<>(); 1483 for (Method m : interfaceType.getMethods()) 1484 { 1485 final MethodSignature methodSignature = new MethodSignature(m); 1486 final MethodDescription newMethodDescription = new MethodDescription(m); 1487 if (!map.containsKey(methodSignature)) 1488 { 1489 map.put(methodSignature, newMethodDescription); 1490 } 1491 else 1492 { 1493 if (newMethodDescription.checkedExceptionTypes != null && newMethodDescription.checkedExceptionTypes.length > 0) 1494 { 1495 final MethodDescription methodDescription = map.get(methodSignature); 1496 final Set<String> checkedExceptionTypes = new HashSet<>(); 1497 checkedExceptionTypes.addAll(Arrays.asList(methodDescription.checkedExceptionTypes)); 1498 checkedExceptionTypes.addAll(Arrays.asList(newMethodDescription.checkedExceptionTypes)); 1499 map.put(methodSignature, new MethodDescription( 1500 methodDescription, 1501 checkedExceptionTypes.toArray(new String[checkedExceptionTypes.size()]))); 1502 } 1503 } 1504 } 1505 return map; 1506 } 1507 1508 final private static class MethodSignature implements Comparable<MethodSignature> { 1509 final private Method method; 1510 final private String name; 1511 final private Class<?>[] parameterTypes; 1512 1513 public MethodSignature(Method method) { 1514 this.method = method; 1515 this.name = method.getName(); 1516 this.parameterTypes = method.getParameterTypes(); 1517 } 1518 1519 @Override 1520 public int hashCode() { 1521 final int prime = 31; 1522 int result = 1; 1523 result = prime * result + Arrays.hashCode(parameterTypes); 1524 result = prime * result + ((name == null) ? 0 : name.hashCode()); 1525 return result; 1526 } 1527 1528 @Override 1529 public boolean equals(Object obj) { 1530 if (this == obj) return true; 1531 if (obj == null) return false; 1532 if (getClass() != obj.getClass()) return false; 1533 1534 MethodSignature other = (MethodSignature) obj; 1535 if (!Arrays.equals(parameterTypes, other.parameterTypes)) return false; 1536 1537 return name == null ? other.name == null : name.equals(other.name); 1538 } 1539 1540 @Override 1541 public int compareTo(MethodSignature o) { 1542 return method.getName().compareTo(o.method.getName()); 1543 } 1544 } 1545 1546 @Override 1547 public PlasticClass addToString(final String toStringValue) 1548 { 1549 check(); 1550 1551 if (!isMethodImplemented(PlasticUtils.TO_STRING_DESCRIPTION)) 1552 { 1553 introduceMethod(PlasticUtils.TO_STRING_DESCRIPTION, new InstructionBuilderCallback() 1554 { 1555 @Override 1556 public void doBuild(InstructionBuilder builder) 1557 { 1558 builder.loadConstant(toStringValue).returnResult(); 1559 } 1560 }); 1561 } 1562 1563 return this; 1564 } 1565 1566 @Override 1567 public boolean isMethodImplemented(MethodDescription description) 1568 { 1569 return inheritanceData.isImplemented(description.methodName, nameCache.toDesc(description)); 1570 } 1571 1572 @Override 1573 public boolean isInterfaceImplemented(Class interfaceType) 1574 { 1575 assert interfaceType != null; 1576 assert interfaceType.isInterface(); 1577 1578 String interfaceName = nameCache.toInternalName(interfaceType); 1579 1580 return inheritanceData.isInterfaceImplemented(interfaceName); 1581 } 1582 1583 @Override 1584 public String getSuperClassName() 1585 { 1586 return superClassName; 1587 } 1588 1589 @Override 1590 public PlasticClass onConstruct(ConstructorCallback callback) 1591 { 1592 check(); 1593 1594 assert callback != null; 1595 1596 constructorCallbacks.add(callback); 1597 1598 return this; 1599 } 1600 1601 void redirectFieldWrite(String fieldName, boolean privateField, MethodNode method) 1602 { 1603 FieldInstrumentation fi = new FieldInstrumentation(method.name, method.desc); 1604 1605 fieldInstrumentations.write.put(fieldName, fi); 1606 1607 if (!proxy) 1608 { 1609 pool.setFieldWriteInstrumentation(classNode.name, fieldName, fi); 1610 } 1611 } 1612 1613 void redirectFieldRead(String fieldName, boolean privateField, MethodNode method) 1614 { 1615 FieldInstrumentation fi = new FieldInstrumentation(method.name, method.desc); 1616 1617 fieldInstrumentations.read.put(fieldName, fi); 1618 1619 if (!proxy) 1620 { 1621 pool.setFieldReadInstrumentation(classNode.name, fieldName, fi); 1622 } 1623 } 1624 1625 final private MethodComparator METHOD_COMPARATOR = new MethodComparator(); 1626 1627 final private class MethodComparator implements Comparator<Method> 1628 { 1629 1630 @Override 1631 public int compare(Method o1, Method o2) 1632 { 1633 1634 int comparison = comparison = o1.getName().compareTo(o2.getName()); 1635 1636 if (comparison == 0) 1637 { 1638 comparison = o1.getParameterTypes().length - o2.getParameterTypes().length; 1639 } 1640 1641 if (comparison == 0) 1642 { 1643 final int count = o1.getParameterTypes().length; 1644 for (int i = 0; i < count; i++) 1645 { 1646 Class p1 = o1.getParameterTypes()[i]; 1647 Class p2 = o1.getParameterTypes()[i]; 1648 if (!p1.equals(p2)) 1649 { 1650 comparison = p1.getName().compareTo(p2.getName()); 1651 break; 1652 } 1653 } 1654 } 1655 1656 if (comparison == 0) 1657 { 1658 // Trying to get methods lower in the interface hierarchy sorted before the same methods 1659 // higher in the hierarchy 1660 Class<?> declaringClass1 = o1.getDeclaringClass(); 1661 Class<?> declaringClass2 = o2.getDeclaringClass(); 1662 if (declaringClass1.isInterface() && declaringClass2.isInterface() && 1663 !declaringClass1.equals(declaringClass2)) { 1664 if (declaringClass1.isAssignableFrom(declaringClass2)) 1665 { 1666 comparison = -1; 1667 } 1668 else if (declaringClass1.isAssignableFrom(declaringClass2)) 1669 { 1670 comparison = 1; 1671 } 1672 } 1673 } 1674 1675 return comparison; 1676 } 1677 } 1678 1679 private List<Method> getUniqueMethods(Class interfaceType) 1680 { 1681 final List<Method> unique = new ArrayList<>(Arrays.asList(interfaceType.getMethods())); 1682 Collections.sort(unique, METHOD_COMPARATOR); 1683 Method last = null; 1684 Iterator<Method> iterator = unique.iterator(); 1685 while (iterator.hasNext()) 1686 { 1687 Method m = iterator.next(); 1688 if (last != null && METHOD_COMPARATOR.compare(m, last) == 0) 1689 { 1690 last = m; 1691 iterator.remove(); 1692 } 1693 } 1694 return unique; 1695 } 1696 1697 @Override 1698 public String toString() 1699 { 1700 return String.format("PlasticClassImpl[%s]", className); 1701 } 1702}