Open Source Repository

Home /spring/spring-context-3.0.5 | Repository Home



org/springframework/instrument/classloading/oc4j/OC4JClassPreprocessorAdapter.java
/*
 * Copyright 2006-2009 the original author or authors.
 
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 
 *      http://www.apache.org/licenses/LICENSE-2.0
 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.springframework.instrument.classloading.oc4j;

import java.lang.instrument.ClassFileTransformer;
import java.lang.instrument.IllegalClassFormatException;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.security.ProtectionDomain;

/**
 * Adapter that implements OC4J ClassPreProcessor interface, delegating to a
 * standard JDK {@link ClassFileTransformer} underneath.
 *
 <p>To avoid compile time checks again the vendor API, a dynamic proxy is
 * being used.
 
 @author Costin Leau
 */
class OC4JClassPreprocessorAdapter implements InvocationHandler {

  private final ClassFileTransformer transformer;

  /**
   * Creates a new {@link OC4JClassPreprocessorAdapter}.
   @param transformer the {@link ClassFileTransformer} to be adapted (must
   * not be <code>null</code>)
   */
  public OC4JClassPreprocessorAdapter(ClassFileTransformer transformer) {
    this.transformer = transformer;
  }

  public Object invoke(Object proxy, Method method, Object[] argsthrows Throwable {
    String name = method.getName();

    if ("equals".equals(name)) {
      return (Boolean.valueOf(proxy == args[0]));
    else if ("hashCode".equals(name)) {
      return hashCode();
    else if ("toString".equals(name)) {
      return toString();
    else if ("initialize".equals(name)) {
      initialize(proxy, (ClassLoaderargs[0]);
      return null;
    else if ("processClass".equals(name)) {
      return processClass((Stringargs[0](byte[]) args[1](Integerargs[2](Integerargs[3],
          (ProtectionDomainargs[4](ClassLoaderargs[5]);
    else {
      throw new IllegalArgumentException("Unknown method: " + method);
    }
  }

  // maps to oracle.classloader.util.ClassPreprocessor#initialize
  // the proxy is passed since it implements the Oracle interface which
  // is asked as a return type
  public Object initialize(Object proxy, ClassLoader loader) {
    return proxy;
  }

  public byte[] processClass(String className, byte origClassBytes[]int offset, int length, ProtectionDomain pd,
      ClassLoader loader) {
    try {
      byte[] tempArray = new byte[length];
      System.arraycopy(origClassBytes, offset, tempArray, 0, length);

      // NB: OC4J passes className as "." without class while the
      // transformer expects a VM, "/" format
      byte[] result = this.transformer.transform(loader, className.replace('.''/'), null, pd, tempArray);
      return (result != null ? result : origClassBytes);
    catch (IllegalClassFormatException ex) {
      throw new IllegalStateException("Cannot transform because of illegal class format", ex);
    }
  }

  @Override
  public String toString() {
    StringBuilder builder = new StringBuilder(getClass().getName());
    builder.append(" for transformer: ");
    builder.append(this.transformer);
    return builder.toString();
  }
}