- /*
- * @(#)CorbaConnectionCacheBase.java 1.25 04/06/21
- *
- * Copyright 2004 Sun Microsystems, Inc. All rights reserved.
- * SUN PROPRIETARY/CONFIDENTIAL. Use is subject to license terms.
- */
- package com.sun.corba.se.impl.transport;
- import java.util.Collection;
- import java.util.Iterator;
- import com.sun.corba.se.pept.broker.Broker;
- import com.sun.corba.se.pept.transport.Connection;
- import com.sun.corba.se.pept.transport.ConnectionCache;
- import com.sun.corba.se.spi.logging.CORBALogDomains;
- import com.sun.corba.se.spi.orb.ORB;
- import com.sun.corba.se.spi.transport.CorbaConnectionCache;
- import com.sun.corba.se.impl.logging.ORBUtilSystemException;
- import com.sun.corba.se.impl.orbutil.ORBUtility;
- /**
- * @author Harold Carr
- */
- public abstract class CorbaConnectionCacheBase
- implements
- ConnectionCache,
- CorbaConnectionCache
- {
- protected ORB orb;
- protected long timestamp = 0;
- protected String cacheType;
- protected String monitoringName;
- protected ORBUtilSystemException wrapper;
- protected CorbaConnectionCacheBase(ORB orb, String cacheType,
- String monitoringName)
- {
- this.orb = orb;
- this.cacheType = cacheType;
- this.monitoringName = monitoringName;
- wrapper =ORBUtilSystemException.get(orb,CORBALogDomains.RPC_TRANSPORT);
- registerWithMonitoring();
- dprintCreation();
- }
- ////////////////////////////////////////////////////
- //
- // pept.transport.ConnectionCache
- //
- public String getCacheType()
- {
- return cacheType;
- }
- public synchronized void stampTime(Connection c)
- {
- // _REVISIT_ Need to worry about wrap around some day
- c.setTimeStamp(timestamp++);
- }
- public long numberOfConnections()
- {
- synchronized (backingStore()) {
- return values().size();
- }
- }
- public long numberOfIdleConnections()
- {
- long count = 0;
- synchronized (backingStore()) {
- Iterator connections = values().iterator();
- while (connections.hasNext()) {
- if (! ((Connection)connections.next()).isBusy()) {
- count++;
- }
- }
- }
- return count;
- }
- public long numberOfBusyConnections()
- {
- long count = 0;
- synchronized (backingStore()) {
- Iterator connections = values().iterator();
- while (connections.hasNext()) {
- if (((Connection)connections.next()).isBusy()) {
- count++;
- }
- }
- }
- return count;
- }
- /**
- * Discarding least recently used Connections that are not busy
- *
- * This method must be synchronized since one WorkerThread could
- * be reclaming connections inside the synchronized backingStore
- * block and a second WorkerThread (or a SelectorThread) could have
- * already executed the if (numberOfConnections <= .... ). As a
- * result the second thread would also attempt to reclaim connections.
- *
- * If connection reclamation becomes a performance issue, the connection
- * reclamation could make its own task and consequently executed in
- * a separate thread.
- * Currently, the accept & reclaim are done in the same thread, WorkerThread
- * by default. It could be changed such that the SelectorThread would do
- * it for SocketChannels and WorkerThreads for Sockets by updating the
- * ParserTable.
- */
- synchronized public boolean reclaim()
- {
- try {
- long numberOfConnections = numberOfConnections();
- if (orb.transportDebugFlag) {
- dprint(".reclaim->: " + numberOfConnections
- + " ("
- + orb.getORBData().getHighWaterMark()
- + "/"
- + orb.getORBData().getLowWaterMark()
- + "/"
- + orb.getORBData().getNumberToReclaim()
- + ")");
- }
- if (numberOfConnections <= orb.getORBData().getHighWaterMark() ||
- numberOfConnections < orb.getORBData().getLowWaterMark()) {
- return false;
- }
- Object backingStore = backingStore();
- synchronized (backingStore) {
- // REVISIT - A less expensive alternative connection reclaiming
- // algorithm could be investigated.
- for (int i=0; i < orb.getORBData().getNumberToReclaim(); i++) {
- Connection toClose = null;
- long lru = java.lang.Long.MAX_VALUE;
- Iterator iterator = values().iterator();
- // Find least recently used and not busy connection in cache
- while ( iterator.hasNext() ) {
- Connection c = (Connection) iterator.next();
- if ( !c.isBusy() && c.getTimeStamp() < lru ) {
- toClose = c;
- lru = c.getTimeStamp();
- }
- }
- if ( toClose == null ) {
- return false;
- }
- try {
- if (orb.transportDebugFlag) {
- dprint(".reclaim: closing: " + toClose);
- }
- toClose.close();
- } catch (Exception ex) {
- // REVISIT - log
- }
- }
- if (orb.transportDebugFlag) {
- dprint(".reclaim: connections reclaimed ("
- + (numberOfConnections - numberOfConnections()) + ")");
- }
- }
- // XXX is necessary to do a GC to reclaim
- // closed network connections ??
- // java.lang.System.gc();
- return true;
- } finally {
- if (orb.transportDebugFlag) {
- dprint(".reclaim<-: " + numberOfConnections());
- }
- }
- }
- ////////////////////////////////////////////////////
- //
- // spi.transport.ConnectionCache
- //
- public String getMonitoringName()
- {
- return monitoringName;
- }
- ////////////////////////////////////////////////////
- //
- // Implementation
- //
- // This is public so folb.Server test can access it.
- public abstract Collection values();
- protected abstract Object backingStore();
- protected abstract void registerWithMonitoring();
- protected void dprintCreation()
- {
- if (orb.transportDebugFlag) {
- dprint(".constructor: cacheType: " + getCacheType()
- + " monitoringName: " + getMonitoringName());
- }
- }
- protected void dprintStatistics()
- {
- if (orb.transportDebugFlag) {
- dprint(".stats: "
- + numberOfConnections() + "/total "
- + numberOfBusyConnections() + "/busy "
- + numberOfIdleConnections() + "/idle"
- + " ("
- + orb.getORBData().getHighWaterMark() + "/"
- + orb.getORBData().getLowWaterMark() + "/"
- + orb.getORBData().getNumberToReclaim()
- + ")");
- }
- }
- protected void dprint(String msg)
- {
- ORBUtility.dprint("CorbaConnectionCacheBase", msg);
- }
- }
- // End of file.