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

jogamp.nativewindow.SharedResourceToolkitLock Maven / Gradle / Ivy

/**
 * Copyright 2012 JogAmp Community. All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without modification, are
 * permitted provided that the following conditions are met:
 *
 *    1. Redistributions of source code must retain the above copyright notice, this list of
 *       conditions and the following disclaimer.
 *
 *    2. Redistributions in binary form must reproduce the above copyright notice, this list
 *       of conditions and the following disclaimer in the documentation and/or other materials
 *       provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY JogAmp Community ``AS IS'' AND ANY EXPRESS OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
 * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JogAmp Community OR
 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
 * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 *
 * The views and conclusions contained in the software and documentation are those of the
 * authors and should not be interpreted as representing official policies, either expressed
 * or implied, of JogAmp Community.
 */

package jogamp.nativewindow;

import java.util.Iterator;

import javax.media.nativewindow.ToolkitLock;

import com.jogamp.common.util.LongObjectHashMap;
import com.jogamp.common.util.locks.LockFactory;
import com.jogamp.common.util.locks.RecursiveLock;

/**
 * Implementing a shared resource based recursive {@link javax.media.nativewindow.ToolkitLock}.
 * 

* A resource handle maybe used within many objects * and can be synchronized across threads via an unique instance of SharedResourceToolkitLock. *

*

* Implementation holds a synchronized map from handle to reference counted {@link SharedResourceToolkitLock}. * New elements are added via {@link #get(long)} if new * and removed via {@link #dispose()} if no more referenced. *

*/ public class SharedResourceToolkitLock implements ToolkitLock { private static final LongObjectHashMap handle2Lock; static { handle2Lock = new LongObjectHashMap(); handle2Lock.setKeyNotFoundValue(null); } /** * @return number of unclosed EGL Displays.
*/ public static int shutdown(boolean verbose) { if(DEBUG || verbose || handle2Lock.size() > 0 ) { System.err.println("SharedResourceToolkitLock: Shutdown (open: "+handle2Lock.size()+")"); if(DEBUG) { Thread.dumpStack(); } if( handle2Lock.size() > 0) { dumpOpenDisplayConnections(); } } return handle2Lock.size(); } public static void dumpOpenDisplayConnections() { System.err.println("SharedResourceToolkitLock: Open ResourceToolkitLock's: "+handle2Lock.size()); int i=0; for(Iterator iter = handle2Lock.iterator(); iter.hasNext(); i++) { final LongObjectHashMap.Entry e = iter.next(); System.err.println("SharedResourceToolkitLock: Open["+i+"]: "+e.value); } } public static final SharedResourceToolkitLock get(long handle) { SharedResourceToolkitLock res; synchronized(handle2Lock) { res = (SharedResourceToolkitLock) handle2Lock.get(handle); if( null == res ) { res = new SharedResourceToolkitLock(handle); res.refCount++; handle2Lock.put(handle, res); if(DEBUG || TRACE_LOCK) { System.err.println("SharedResourceToolkitLock.get() * NEW *: "+res); } } else { res.refCount++; if(DEBUG || TRACE_LOCK) { System.err.println("SharedResourceToolkitLock.get() * EXIST *: "+res); } } } return res; } private final RecursiveLock lock; private final long handle; private volatile int refCount; private SharedResourceToolkitLock(long handle) { this.lock = LockFactory.createRecursiveLock(); this.handle = handle; this.refCount = 0; } @Override public final void lock() { lock.lock(); if(TRACE_LOCK) { System.err.println("SharedResourceToolkitLock.lock()"); } } @Override public final void unlock() { if(TRACE_LOCK) { System.err.println("SharedResourceToolkitLock.unlock()"); } lock.unlock(); } @Override public final void validateLocked() throws RuntimeException { lock.validateLocked(); } @Override public final void dispose() { if(0 < refCount) { // volatile OK synchronized(handle2Lock) { refCount--; if(0 == refCount) { if(DEBUG || TRACE_LOCK) { System.err.println("SharedResourceToolkitLock.dispose() * REMOV *: "+this); } handle2Lock.remove(handle); } else { if(DEBUG || TRACE_LOCK) { System.err.println("SharedResourceToolkitLock.dispose() * DOWN *: "+this); } } } } else { if(DEBUG || TRACE_LOCK) { System.err.println("SharedResourceToolkitLock.dispose() * NULL *: "+this); } } } @Override public String toString() { return "SharedResourceToolkitLock[refCount "+refCount+", handle 0x"+Long.toHexString(handle)+", obj 0x"+Integer.toHexString(hashCode())+", isOwner "+lock.isOwner(Thread.currentThread())+", "+lock.toString()+"]"; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy