com.kenai.jffi.CallContextCache Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of driver-cql-shaded Show documentation
Show all versions of driver-cql-shaded Show documentation
A Shaded CQL ActivityType driver for http://nosqlbench.io/
/*
* Copyright (C) 2009 Wayne Meissner
*
* This file is part of jffi.
*
* 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.
*
*
* Alternatively, you can redistribute it and/or modify it under
* the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This code 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. See the GNU Lesser General Public License
* for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this work. If not, see .
*/
package com.kenai.jffi;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.SoftReference;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
public class CallContextCache {
private final Map contextCache = new ConcurrentHashMap();
private final ReferenceQueue contextReferenceQueue = new ReferenceQueue();
/** Holder class to do lazy allocation of the ClosureManager instance */
private static final class SingletonHolder {
static final CallContextCache INSTANCE = new CallContextCache();
}
/**
* Gets the global instance of the CallContextCache
*
* @return An instance of a CallContextCache
*/
public static CallContextCache getInstance() {
return SingletonHolder.INSTANCE;
}
/** Constructs a ClosureManager */
private CallContextCache() { }
public final CallContext getCallContext(Type returnType, Type[] parameterTypes, CallingConvention convention) {
return getCallContext(returnType, parameterTypes, convention, true, false);
}
public final CallContext getCallContext(Type returnType, Type[] parameterTypes, CallingConvention convention,
boolean saveErrno) {
return getCallContext(returnType, parameterTypes, convention, saveErrno, false);
}
public final CallContext getCallContext(Type returnType, Type[] parameterTypes, CallingConvention convention,
boolean saveErrno, boolean faultProtect) {
Signature signature = new Signature(returnType, parameterTypes, convention, saveErrno, faultProtect);
CallContextRef ref = contextCache.get(signature);
CallContext ctx;
if (ref != null && (ctx = ref.get()) != null) {
return ctx;
}
// Cull any dead references
while ((ref = (CallContextRef) contextReferenceQueue.poll()) != null) {
contextCache.remove(ref.signature);
}
ctx = new CallContext(returnType, parameterTypes.clone(), convention, saveErrno, faultProtect);
contextCache.put(signature, new CallContextRef(signature, ctx, contextReferenceQueue));
return ctx;
}
private static final class CallContextRef extends SoftReference {
final Signature signature;
public CallContextRef(Signature signature, CallContext ctx, ReferenceQueue queue) {
super(ctx, queue);
this.signature = signature;
}
}
private static final class Signature {
/**
* Keep references to the return and parameter types so they do not get
* garbage collected until the closure does.
*/
private final Type returnType;
private final Type[] parameterTypes;
private final CallingConvention convention;
private final boolean saveErrno;
private final boolean faultProtect;
private int hashCode = 0;
public Signature(Type returnType, Type[] parameterTypes, CallingConvention convention, boolean saveErrno,
boolean faultProtect) {
if (returnType == null || parameterTypes == null) {
throw new NullPointerException("null return type or parameter types array");
}
this.returnType = returnType;
this.parameterTypes = parameterTypes;
this.convention = convention;
this.saveErrno = saveErrno;
this.faultProtect = faultProtect;
}
@Override
public boolean equals(Object obj) {
if (obj == null || getClass() != obj.getClass()) {
return false;
}
final Signature other = (Signature) obj;
if (convention != other.convention || saveErrno != other.saveErrno || faultProtect != other.faultProtect) {
return false;
}
if (this.returnType != other.returnType && !this.returnType.equals(other.returnType)) {
return false;
}
if (this.parameterTypes.length == other.parameterTypes.length) {
for (int i = 0; i < this.parameterTypes.length; ++i) {
if (this.parameterTypes[i] != other.parameterTypes[i] && (this.parameterTypes[i] == null || !this.parameterTypes[i].equals(other.parameterTypes[i]))) {
return false;
}
}
// All param types are same, return type is same, convention is same, so this is the same signature
return true;
}
return false;
}
private final int calculateHashCode() {
int hash = 7;
hash = 53 * hash + (this.returnType != null ? this.returnType.hashCode() : 0);
int paramHash = 1;
for (int i = 0; i < parameterTypes.length; ++i) {
paramHash = 31 * paramHash + parameterTypes[i].hashCode();
}
hash = 53 * hash + paramHash;
hash = 53 * hash + this.convention.hashCode();
hash = 53 * hash + (this.saveErrno ? 1 : 0);
hash = 53 * hash + (this.faultProtect ? 1 : 0);
return hash;
}
@Override
public int hashCode() {
return hashCode != 0 ? hashCode : (hashCode = calculateHashCode());
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy