/* * Copyright (c) 1998-2011 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 Reza Rahman */ package com.caucho.config.gen; import java.io.IOException; import java.util.HashMap; import java.util.concurrent.TimeUnit; import javax.ejb.LockType; import javax.enterprise.inject.spi.AnnotatedMethod; import com.caucho.inject.Module; import com.caucho.java.JavaWriter; /** * Represents EJB lock type specification interception. The specification gears * it towards EJB singletons, but it can be used for other bean types. */ @Module public class LockGenerator<X> extends AbstractAspectGenerator<X> { private static final int DEFAULT_TIMEOUT = 10000; private boolean _isContainerManaged; private LockType _lockType; private long _lockTimeout; private TimeUnit _lockTimeoutUnit; public LockGenerator(LockFactory<X> factory, AnnotatedMethod<? super X> method, AspectGenerator<X> next, LockType lockType, long lockTimeout, TimeUnit lockTimeoutUnit) { super(factory, method, next); _isContainerManaged = true; _lockType = lockType; if (lockTimeoutUnit != null) { _lockTimeout = lockTimeout; _lockTimeoutUnit = lockTimeoutUnit; } else { _lockTimeout = DEFAULT_TIMEOUT; _lockTimeoutUnit = TimeUnit.MILLISECONDS; } } /** * Generates the class prologue. */ @Override public void generateMethodPrologue(JavaWriter out, HashMap<String,Object> map) throws IOException { if ((_isContainerManaged && (_lockType != null)) && (map.get("caucho.ejb.lock") == null)) { map.put("caucho.ejb.lock", "done"); // TCK: fairness flag must be true for the tck out.println(); out.println("private transient final java.util.concurrent.locks.ReentrantReadWriteLock _readWriteLock"); out.println(" = new java.util.concurrent.locks.ReentrantReadWriteLock(true);"); } super.generateMethodPrologue(out, map); } /** * Generates the method interception code. */ @Override public void generatePreTry(JavaWriter out) throws IOException { out.println("boolean isLocked = false;"); if (_isContainerManaged && (_lockType != null)) { out.println(); switch (_lockType) { case READ: if (_lockTimeout != -1) { out.println("com.caucho.config.util.LockUtil.lockRead(" + "_readWriteLock.readLock()," + _lockTimeoutUnit.toMillis(_lockTimeout) + "L" + ");"); } else { // XXX: This should probably be put behind the lock utility as well, // mostly to maintain code symmetry. out.println("_readWriteLock.readLock().lock();"); } break; case WRITE: if (_lockTimeout != -1) { out.println("com.caucho.config.util.LockUtil.lockWrite(" + "_readWriteLock," + _lockTimeoutUnit.toMillis(_lockTimeout) + "L" + ");"); } else { out.println("com.caucho.config.util.LockUtil.lockWrite(" + "_readWriteLock);"); } break; } out.println(); out.println("try {"); out.pushDepth(); } out.println("isLocked = true;"); super.generatePreTry(out); } /** * Generates the method interception code. */ @Override public void generatePostFinally(JavaWriter out) throws IOException { super.generatePostFinally(out); if (_isContainerManaged && (_lockType != null)) { out.popDepth(); out.println("} finally {"); out.pushDepth(); switch (_lockType) { case READ: out.println(); out.println("if (isLocked)"); out.println(" _readWriteLock.readLock().unlock();"); break; case WRITE: out.println(); out.println("if (isLocked)"); out.println(" _readWriteLock.writeLock().unlock();"); break; } out.popDepth(); out.println("}"); } } }