Package org.apache.derby.impl.services.locks

Source Code of org.apache.derby.impl.services.locks.LockSet

/*

   Derby - Class org.apache.derby.impl.services.locks.LockSet

   Licensed to the Apache Software Foundation (ASF) under one or more
   contributor license agreements.  See the NOTICE file distributed with
   this work for additional information regarding copyright ownership.
   The ASF licenses this file to you 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.apache.derby.impl.services.locks;

import org.apache.derby.iapi.services.locks.CompatibilitySpace;
import org.apache.derby.iapi.services.locks.Latch;
import org.apache.derby.iapi.services.locks.Lockable;
import org.apache.derby.iapi.services.locks.C_LockFactory;

import org.apache.derby.iapi.error.StandardException;

import org.apache.derby.iapi.services.sanity.SanityManager;
import org.apache.derby.iapi.services.diag.DiagnosticUtil;

import org.apache.derby.iapi.reference.Property;
import org.apache.derby.iapi.reference.SQLState;

import java.util.HashMap;
import java.util.Enumeration;
import java.util.Iterator;
import java.util.Map;


/**
  A LockSet is a complete lock table.  A lock table is a hash table
  keyed by a Lockable and with a LockControl as the data element.

  <P>
  A LockControl contains information about the locks held on a Lockable.

  <BR>
  MT - Mutable - Container Object : All non-private methods of this class are
  thread safe unless otherwise stated by their javadoc comments.

  <BR>
  All searching of
    the hashtable is performed using java synchroization(this).
  <BR>
  The class creates ActiveLock and LockControl objects.
 
  LockControl objects are never passed out of this class, All the methods of
    LockControl are called while being synchronized on this, thus providing the
    single threading that LockControl required.

  Methods of Lockables are only called by this class or LockControl, and
    always while being synchronized on this, thus providing the single
    threading that Lockable requires.
 
  @see LockControl
*/

final class LockSet implements LockTable {
  /*
  ** Fields
  */
  private final SinglePool factory;

    /** Hash table which maps <code>Lockable</code> objects to
     * <code>Lock</code>s. */
    private final HashMap locks;

  /**
    Timeout for deadlocks, in ms.
    <BR>
    MT - immutable
  */
  private int deadlockTimeout = Property.DEADLOCK_TIMEOUT_DEFAULT * 1000;
  private int waitTimeout = Property.WAIT_TIMEOUT_DEFAULT * 1000;

//EXCLUDE-START-lockdiag-

  // this varible is set and get without synchronization. 
  // Only one thread should be setting it at one time.
  private boolean deadlockTrace;

//EXCLUDE-END-lockdiag-

  // The number of waiters for locks
  private int blockCount;

  /*
  ** Constructor
  */

  protected LockSet(SinglePool factory) {
    this.factory = factory;
    locks = new HashMap();
  }


  /*
  ** Public Methods
  */

  /**
   *  Lock an object within a specific compatibility space.
   *
   *  @param  compatibilitySpace Compatibility space.
   *  @param  ref Lockable reference.
   *  @param  qualifier Qualifier.
   *  @param  timeout Timeout in milli-seconds
   *
   *  @return  Object that represents the lock.
   *
   *  @exception  StandardException Standard Derby policy.

  */
  public Lock lockObject(CompatibilitySpace compatibilitySpace, Lockable ref,
               Object qualifier, int timeout)
    throws StandardException
  {   
    if (SanityManager.DEBUG) {

      if (SanityManager.DEBUG_ON("memoryLeakTrace")) {

        if (locks.size() > 1000)
          System.out.println("memoryLeakTrace:LockSet: " +
                                           locks.size());
      }
    }

    Control gc;
    LockControl control;
    Lock lockItem;
        String  lockDebug = null;

    synchronized (this) {

      gc = getControl(ref);

      if (gc == null) {

        // object is not locked, can be granted
        Lock gl = new Lock(compatibilitySpace, ref, qualifier);

        gl.grant();

        locks.put(ref, gl);

        return gl;
      }

      control = gc.getLockControl();
      if (control != gc) {
        locks.put(ref, control);
      }
     

      if (SanityManager.DEBUG) {
        SanityManager.ASSERT(ref.equals(control.getLockable()));

        // ASSERT item is in the list
                if (getControl(control.getLockable()) != control)
                {
          SanityManager.THROWASSERT(
                        "lockObject mismatched lock items " +
                        getControl(control.getLockable()) + " " + control);
                }
      }

      lockItem = control.addLock(this, compatibilitySpace, qualifier);

      if (lockItem.getCount() != 0) {
        return lockItem;
      }

      if (AbstractPool.noLockWait(timeout, compatibilitySpace)) {

          // remove all trace of lock
          control.giveUpWait(lockItem, this);

                if (SanityManager.DEBUG)
                {
                    if (SanityManager.DEBUG_ON("DeadlockTrace"))
                    {

                        SanityManager.showTrace(new Throwable());

                        // The following dumps the lock table as it
                        // exists at the time a timeout is about to
                        // cause a deadlock exception to be thrown.

                        lockDebug =
                            DiagnosticUtil.toDiagString(lockItem)   +
                            "\nCould not grant lock with zero timeout, here's the table" +
                            this.toDebugString();
                    }
                }

        return null;
      }

    } // synchronized block

    boolean deadlockWait = false;
    int actualTimeout;

    if (timeout == C_LockFactory.WAIT_FOREVER)
    {
      // always check for deadlocks as there should not be any
      deadlockWait = true;
      if ((actualTimeout = deadlockTimeout) == C_LockFactory.WAIT_FOREVER)
        actualTimeout = Property.DEADLOCK_TIMEOUT_DEFAULT * 1000;
    }
    else
    {

      if (timeout == C_LockFactory.TIMED_WAIT)
        timeout = actualTimeout = waitTimeout;
      else
        actualTimeout = timeout;


      // five posible cases
      // i)   timeout -1, deadlock -1         ->
            //          just wait forever, no deadlock check
      // ii)  timeout >= 0, deadlock -1       ->
            //          just wait for timeout, no deadlock check
      // iii) timeout -1, deadlock >= 0       ->
            //          wait for deadlock, then deadlock check,
            //          then infinite timeout
      // iv)  timeout >=0, deadlock < timeout ->
            //          wait for deadlock, then deadlock check,
            //          then wait for (timeout - deadlock)
      // v)   timeout >=0, deadlock >= timeout ->
            //          just wait for timeout, no deadlock check


      if (deadlockTimeout >= 0) {

        if (actualTimeout < 0) {
          // infinite wait but perform a deadlock check first
          deadlockWait = true;
          actualTimeout = deadlockTimeout;
        } else if (deadlockTimeout < actualTimeout) {

          // deadlock wait followed by a timeout wait

          deadlockWait = true;
          actualTimeout = deadlockTimeout;

          // leave timeout as the remaining time
          timeout -= deadlockTimeout;
        }
      }
    }


        ActiveLock waitingLock = (ActiveLock) lockItem;
        lockItem = null;

        int earlyWakeupCount = 0;
        long startWaitTime = 0;

forever:  for (;;) {

                byte wakeupReason = 0;
                ActiveLock nextWaitingLock = null;
                Object[] deadlockData = null;

                try {
                    try {
                        wakeupReason = waitingLock.waitForGrant(actualTimeout);
                    } catch(StandardException e) {
                        // DERBY-4711: If waitForGrant() fails, we need to
                        // remove ourselves from the queue so that those
                        // behind us in the queue don't get stuck waiting for
                        // us.
                        nextWaitingLock = control.getNextWaiter(waitingLock, true, this);
                        throw e;
                    }

                    boolean willQuitWait;
                    Enumeration timeoutLockTable = null;
                    long currentTime = 0;
       
                    synchronized (this) {

                        if (control.isGrantable(
                                control.firstWaiter() == waitingLock,
                                compatibilitySpace,
                                qualifier)) {

                            // Yes, we are granted, put us on the granted queue.
                            control.grant(waitingLock);

                            // Remove from the waiting queue & get next waiter
                            nextWaitingLock =
                                control.getNextWaiter(waitingLock, true, this);

                            return waitingLock;
                        }

                        // try again later
                        waitingLock.clearPotentiallyGranted();

                        willQuitWait =
                            (wakeupReason != Constants.WAITING_LOCK_GRANT);

                        if (((wakeupReason == Constants.WAITING_LOCK_IN_WAIT) &&
                                    deadlockWait) ||
                            (wakeupReason == Constants.WAITING_LOCK_DEADLOCK))
                        {

                            // check for a deadlock, even if we were woken up
                            // because we were selected as a victim we still
                            // check because the situation may have changed.
                            deadlockData =
                                Deadlock.look(
                                    factory, this, control, waitingLock,
                                    wakeupReason);

                            if (deadlockData == null) {
                                // we don't have a deadlock
                                deadlockWait = false;

                                actualTimeout = timeout;
                                startWaitTime = 0;
                                willQuitWait = false;
                            } else {
                                willQuitWait = true;
                            }
                        }

                        nextWaitingLock =
                            control.getNextWaiter(
                                waitingLock, willQuitWait, this);


                        // If we were not woken by another then we have timed
                        // out. Either deadlock out or timeout. Or thread has
                        // been interrupted.
                        if (willQuitWait) {

                            if (SanityManager.DEBUG)
                            {
                                if (SanityManager.DEBUG_ON("DeadlockTrace"))
                                {

                                    SanityManager.showTrace(new Throwable());

                                    // The following dumps the lock table as it
                                    // exists at the time a timeout is about to
                                    // cause a deadlock exception to be thrown.

                                    lockDebug =
                                    DiagnosticUtil.toDiagString(waitingLock)   +
                                    "\nGot deadlock/timeout, here's the table" +
                                    this.toDebugString();
                                }
                            }
                           
                            if (deadlockTrace && (deadlockData == null))
                            {
                                // if ending lock request due to lock timeout
                                // want a copy of the LockTable and the time,
                                // in case of deadlock deadlockData has the
                                // info we need.
                                currentTime = System.currentTimeMillis();
                                timeoutLockTable =
                                    factory.makeVirtualLockTable();
                            }
                        }

                    } // synchronized block

                    // need to do this outside of the synchronized block as the
                    // message text building (timeouts and deadlocks) may
                    // involve getting locks to look up table names from
                    // identifiers.

                    if (willQuitWait)
                    {
                        if (SanityManager.DEBUG)
                        {
                            if (lockDebug != null)
                            {
                                String type =
                                    ((deadlockData != null) ?
                                         "deadlock:" : "timeout:");

                                SanityManager.DEBUG_PRINT(
                                    type,
                                    "wait on lockitem caused " + type +
                                    lockDebug);
                            }

                        }

                       
                        if (deadlockData == null)
                        {
                            // ending wait because of lock timeout or interrupt
                            if (wakeupReason ==
                                    Constants.WAITING_LOCK_INTERRUPTED) {

                                Thread.currentThread().interrupt();

                                throw StandardException.
                                    newException(SQLState.CONN_INTERRUPT);

                            } else if (deadlockTrace)
                            {  
                                // Turn ON derby.locks.deadlockTrace to build
                                // the lockTable.
                                   
                               
                                throw Timeout.buildException(
                                    waitingLock, timeoutLockTable, currentTime);
                            }
                            else
                            {
                                StandardException se =
                                    StandardException.newException(
                                        SQLState.LOCK_TIMEOUT);

                                throw se;
                            }
                        }
                        else
                        {
                            // ending wait because of lock deadlock.

                            throw Deadlock.buildException(
                                    factory, deadlockData);
                        }
                    }
                } finally {
                    if (nextWaitingLock != null) {
                        nextWaitingLock.wakeUp(Constants.WAITING_LOCK_GRANT);
                        nextWaitingLock = null;
                    }
                }

                if (actualTimeout != C_LockFactory.WAIT_FOREVER) {

                    if (wakeupReason != Constants.WAITING_LOCK_IN_WAIT)
                        earlyWakeupCount++;

                    if (earlyWakeupCount > 5) {

                        long now = System.currentTimeMillis();

                        if (startWaitTime != 0) {

                            long sleepTime = now - startWaitTime;

                            actualTimeout -= sleepTime;
                        }

                        startWaitTime = now;
                    }
                }


            } // for(;;)
  }

  /**
    Unlock an object, previously locked by lockObject().

    If unlockCOunt is not zero then the lock will be unlocked
    that many times, otherwise the unlock count is taken from
    item.

  */
  public void unlock(Latch item, int unlockCount) {

    if (SanityManager.DEBUG) {
      if (SanityManager.DEBUG_ON(Constants.LOCK_TRACE)) {
        /*
        ** I don't like checking the trace flag twice, but SanityManager
        ** doesn't provide a way to get to the debug trace stream
        ** directly.
        */
        SanityManager.DEBUG(
                    Constants.LOCK_TRACE,
                    "Release lock: " + DiagnosticUtil.toDiagString(item));
      }
    }

    boolean tryGrant = false;
    ActiveLock nextGrant = null;

    synchronized (this) {

      Control control = getControl(item.getLockable());
     
      if (SanityManager.DEBUG) {

                // only valid Lock's expected
                if (item.getLockable() == null)
                {
                    SanityManager.THROWASSERT(
                        "item.getLockable() = null." +
                        "unlockCount " + unlockCount +
                        "item = " + DiagnosticUtil.toDiagString(item));
                }

                // only valid Lock's expected
                if (control == null)
                {
                    SanityManager.THROWASSERT(
                        "control = null." +
                        "unlockCount " + unlockCount +
                        "item = " + DiagnosticUtil.toDiagString(item));
                }

        if (getControl(control.getLockable()) != control)
                {
                    SanityManager.THROWASSERT(
                        "unlock mismatched lock items " +
                        getControl(control.getLockable()) + " " + control);
                }

        if ((unlockCount != 0) && (unlockCount > item.getCount()))
          SanityManager.THROWASSERT("unlockCount " + unlockCount +
            " larger than actual lock count " + item.getCount() + " item " + item);
      }

      tryGrant = control.unlock(item, unlockCount);
      item = null;

      boolean mayBeEmpty = true;
      if (tryGrant) {
        nextGrant = control.firstWaiter();
        if (nextGrant != null) {
          mayBeEmpty = false;
          if (!nextGrant.setPotentiallyGranted())
            nextGrant = null;
        }
      }

      if (mayBeEmpty) {
        if (control.isEmpty()) {
          // no-one granted, no-one waiting, remove lock control
          locks.remove(control.getLockable());
        }
        return;
      }
    } // synchronized (this)

    if (tryGrant && (nextGrant != null)) {
      nextGrant.wakeUp(Constants.WAITING_LOCK_GRANT);
    }
  }

    /**
     * Unlock an object once if it is present in the specified group. Also
     * remove the object from the group.
     *
     * @param space the compatibility space
     * @param ref a reference to the locked object
     * @param qualifier qualifier of the lock
     * @param group a map representing the locks in a group
     * @return the corresponding lock in the group map, or <code>null</code> if
     * the object was not unlocked
     */
    public synchronized Lock unlockReference(CompatibilitySpace space,
                                             Lockable ref, Object qualifier,
                                             Map group) {

        Control control = getControl(ref);
        if (control == null) {
            return null;
        }

        Lock setLock = control.getLock(space, qualifier);
        if (setLock == null) {
            return null;
        }

        Lock lockInGroup = (Lock) group.remove(setLock);
        if (lockInGroup != null) {
            unlock(lockInGroup, 1);
        }

        return lockInGroup;
    }

    /**
     * {@inheritDoc}
     */
    public boolean zeroDurationLockObject(
        CompatibilitySpace space, Lockable ref, Object qualifier, int timeout)
            throws StandardException {

        if (SanityManager.DEBUG) {
            if (SanityManager.DEBUG_ON(Constants.LOCK_TRACE)) {
                D_LockControl.debugLock(
                    "Zero Duration Lock Request before Grant: ",
                    space, null, ref, qualifier, timeout);
                if (SanityManager.DEBUG_ON(Constants.LOCK_STACK_TRACE)) {
                    // The following will print the stack trace of the lock
                    // request to the log.
                    Throwable t = new Throwable();
                    java.io.PrintWriter istream =
                        SanityManager.GET_DEBUG_STREAM();
                    istream.println("Stack trace of lock request:");
                    t.printStackTrace(istream);
                }
            }
        }

        // Very fast zeroDurationLockObject() for unlocked objects.
        // If no entry exists in the lock manager for this reference
        // then it must be unlocked.
        // If the object is locked then we perform a grantable
        // check, skipping over any waiters.
        // If the caller wants to wait and the lock cannot
        // be granted then we do the slow join the queue and
        // release the lock method.

        synchronized (this) {
            Control control = getControl(ref);
            if (control == null) {
                return true;
            }

            // If we are grantable, ignoring waiting locks then
            // we can also grant this request now, as skipping
            // over the waiters won't block them as we release
            // the lock rightway.
            if (control.isGrantable(true, space, qualifier)) {
                return true;
            }

            // can't be granted and are not willing to wait.
            if (AbstractPool.noLockWait(timeout, space)) {
                return false;
            }
        }

        Lock lock = lockObject(space, ref, qualifier, timeout);

        if (SanityManager.DEBUG) {
            if (SanityManager.DEBUG_ON(Constants.LOCK_TRACE)) {
                D_LockControl.debugLock(
                    "Zero Lock Request Granted: ",
                    space, null, ref, qualifier, timeout);
            }
        }

        // and simply unlock it once
        unlock(lock, 1);

        return true;
    }

    /**
     * Set the deadlock timeout.
     *
     * @param timeout deadlock timeout in milliseconds
     */
    public void setDeadlockTimeout(int timeout) {
        deadlockTimeout = timeout;
    }

    /**
     * Set the wait timeout.
     *
     * @param timeout wait timeout in milliseconds
     */
    public void setWaitTimeout(int timeout) {
        waitTimeout = timeout;
    }
 
    /**
     * Get the wait timeout in milliseconds.
     */
    public int getWaitTimeout() { return waitTimeout; }
   
  /*
  ** Non public methods
  */
//EXCLUDE-START-lockdiag-

  public void setDeadlockTrace(boolean val)
  {
    // set this without synchronization
    deadlockTrace = val;
  }     
//EXCLUDE-END-lockdiag-

    public String toDebugString()
    {
        if (SanityManager.DEBUG)
        {
            String str = new String();

            int i = 0;
            for (Iterator it = locks.values().iterator(); it.hasNext(); )
            {
                str += "\n  lock[" + i + "]: " +
                    DiagnosticUtil.toDiagString(it.next());
            }

            return(str);
        }
        else
        {
            return(null);
        }
    }

    /**
     * Add all waiters in this lock table to a <code>Map</code> object.
     * <br>
     * MT - must be synchronized on this <code>LockSet</code> object.
     */
    public void addWaiters(Map waiters) {
        for (Iterator it = locks.values().iterator(); it.hasNext(); ) {
            Control control = (Control) it.next();
            control.addWaiters(waiters);
        }
    }

//EXCLUDE-START-lockdiag-
  /**
   * make a shallow clone of myself and my lock controls
   */
  public synchronized Map shallowClone()
  {
    HashMap clone = new HashMap();

    for (Iterator it = locks.keySet().iterator(); it.hasNext(); )
    {
      Lockable lockable = (Lockable) it.next();
      Control control = getControl(lockable);

      clone.put(lockable, control.shallowClone());
    }

    return clone;
  }
//EXCLUDE-END-lockdiag-

  /*
  ** Support for anyoneBlocked(). These methods assume that caller
  ** is synchronized on this LockSet object.
  */

  /**
   * Increase blockCount by one.
   * <BR>
   * MT - must be synchronized on this <code>LockSet</code> object.
   */
  public void oneMoreWaiter() {
    blockCount++;
  }

  /**
   * Decrease blockCount by one.
   * <BR>
   * MT - must be synchronized on this <code>LockSet</code> object.
   */
  public void oneLessWaiter() {
    blockCount--;
  }

  public synchronized boolean anyoneBlocked() {
    if (SanityManager.DEBUG) {
      SanityManager.ASSERT(
        blockCount >= 0, "blockCount should not be negative");
    }

    return blockCount != 0;
  }

  /**
   * Get the <code>Control</code> for an object in the lock table.
   * <br>
   * MT - must be synchronized on this <code>LockSet</code> object.
   */
  private final Control getControl(Lockable ref) {
    return (Control) locks.get(ref);
  }
}
TOP

Related Classes of org.apache.derby.impl.services.locks.LockSet

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.