All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.caucho.loader.CompilingClassEntry Maven / Gradle / Ivy

There is a newer version: 4.0.66
Show newest version
/*
 * Copyright (c) 1998-2012 Caucho Technology -- all rights reserved
 *
 * This file is part of Resin(R) Open Source
 *
 * Each copy or derived work must preserve the copyright notice and this
 * notice unmodified.
 *
 * Resin Open Source is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * Resin Open Source is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty
 * of NON-INFRINGEMENT.  See the GNU General Public License for more
 * details.
 *
 * You should have received a copy of the GNU General Public License
 * along with Resin Open Source; if not, write to the
 *
 *   Free Software Foundation, Inc.
 *   59 Temple Place, Suite 330
 *   Boston, MA 02111-1307  USA
 *
 * @author Scott Ferguson
 */

package com.caucho.loader;

import com.caucho.env.thread.ThreadPool;
import com.caucho.java.CompileClassNotFound;
import com.caucho.server.util.CauchoSystem;
import com.caucho.util.L10N;
import com.caucho.vfs.Path;

import java.io.IOException;
import java.security.CodeSource;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.logging.Logger;
import java.util.logging.Level;

class CompilingClassEntry extends ClassEntry {
  private static final L10N L = new L10N(CompilingClassEntry.class);
  private static final Logger log
    = Logger.getLogger(CompilingClassEntry.class.getName());
  
  private CompilingLoader _loader;
  private boolean _compileIsModified;
  private AtomicBoolean _isCompiling = new AtomicBoolean();
    
  public CompilingClassEntry(CompilingLoader compilingLoader,
                             DynamicClassLoader loader,
                             String name, Path sourcePath,
                             Path classPath,
                             CodeSource codeSource)
  {
    super(loader, name, sourcePath, classPath, codeSource);

    _loader = compilingLoader;
  }

  @Override
  public void preLoad()
    throws ClassNotFoundException
  {
    String javaName = getName().replace('.', '/') + ".java";
    
    Path javaFile = getSourcePath();
    Path classFile = getClassPath();
    
    if (javaFile.getLastModified() <= classFile.getLastModified()) {
      log.finest(L.l("loading pre-compiled class {0} from {1}",
                     getName(), classFile));
      return;
    }

    if (! javaFile.canRead())
      return;
    
    try {
      classFile.remove();
    } catch (IOException e) {
    }
      
    String sourcePath = _loader.prefixClassPath(_loader.getClassLoader().getSourcePath());

    // deal with windows case nuttiness
    if (CauchoSystem.isWindows()
        && ! _loader.checkSource(_loader.getSource(), javaName))
      return;

    _loader.compileClass(javaFile, classFile, sourcePath, false);

    if (classFile.canRead()) {
      log.fine(L.l("loading compiled class {0} from {1}",
                   getName(), classFile));
    }
    else if (javaFile.exists())
      throw new CompileClassNotFound(L.l("{1} does not have a class file because compiling {0} didn't produce a {1} class",
                                         javaFile, getName()));

    setDependPath(classFile);
  }

  /**
   * Returns true if the compile doesn't avoid the dependency.
   */
  public boolean compileIsModified()
  {
    if (_compileIsModified)
      return true;

    CompileThread compileThread = new CompileThread();
    ThreadPool.getThreadPool().start(compileThread);

    try {
      synchronized (compileThread) {
        if (! compileThread.isDone())
          compileThread.wait(5000);
      }

      if (_compileIsModified)
        return true;
      else if (compileThread.isDone()) {
        setDependPath(getClassPath());

        return reloadIsModified();
      }
      else
        return true;
    } catch (Throwable e) {
    }

    return false;
  }

  class CompileThread implements Runnable {
    private volatile boolean _isDone;

    public boolean isDone()
    {
      return _isDone;
    }
    
    public void run()
    {
      Path sourcePath = getSourcePath();
      
      long length = sourcePath.getLength();
      long lastModified = sourcePath.getLastModified();

      try {
        _loader.compileClass(getSourcePath(), getClassPath(),
                             getSourcePath().getPath(), false);

        setSourceLength(length);
        setSourceLastModified(lastModified);
      } catch (Throwable e) {
        log.log(Level.FINE, e.toString(), e);

        _compileIsModified = true;
      }

      _isDone = true;

      synchronized (this) {
        notifyAll();
      }
    }
  }

  public String toString()
  {
    return getClass().getSimpleName() + "[" + getClassPath() + ", src=" + getSourcePath() + "]";
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy