mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			
		
			
				
	
	
		
			398 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Java
		
	
	
	
			
		
		
	
	
			398 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Java
		
	
	
	
| /* SelectorImpl.java --
 | |
|    Copyright (C) 2002, 2003, 2004, 2006  Free Software Foundation, Inc.
 | |
| 
 | |
| This file is part of GNU Classpath.
 | |
| 
 | |
| GNU Classpath 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, or (at your option)
 | |
| any later version.
 | |
| 
 | |
| GNU Classpath 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
 | |
| General Public License for more details.
 | |
| 
 | |
| You should have received a copy of the GNU General Public License
 | |
| along with GNU Classpath; see the file COPYING.  If not, write to the
 | |
| Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
 | |
| 02110-1301 USA.
 | |
| 
 | |
| Linking this library statically or dynamically with other modules is
 | |
| making a combined work based on this library.  Thus, the terms and
 | |
| conditions of the GNU General Public License cover the whole
 | |
| combination.
 | |
| 
 | |
| As a special exception, the copyright holders of this library give you
 | |
| permission to link this library with independent modules to produce an
 | |
| executable, regardless of the license terms of these independent
 | |
| modules, and to copy and distribute the resulting executable under
 | |
| terms of your choice, provided that you also meet, for each linked
 | |
| independent module, the terms and conditions of the license of that
 | |
| module.  An independent module is a module which is not derived from
 | |
| or based on this library.  If you modify this library, you may extend
 | |
| this exception to your version of the library, but you are not
 | |
| obligated to do so.  If you do not wish to do so, delete this
 | |
| exception statement from your version. */
 | |
| 
 | |
| 
 | |
| package gnu.java.nio;
 | |
| 
 | |
| import java.io.IOException;
 | |
| import java.nio.channels.ClosedSelectorException;
 | |
| import java.nio.channels.SelectableChannel;
 | |
| import java.nio.channels.SelectionKey;
 | |
| import java.nio.channels.Selector;
 | |
| import java.nio.channels.SocketChannel;
 | |
| import java.nio.channels.spi.AbstractSelectableChannel;
 | |
| import java.nio.channels.spi.AbstractSelector;
 | |
| import java.nio.channels.spi.SelectorProvider;
 | |
| import java.util.Collections;
 | |
| import java.util.HashSet;
 | |
| import java.util.Iterator;
 | |
| import java.util.Set;
 | |
| 
 | |
| public class SelectorImpl extends AbstractSelector
 | |
| {
 | |
|   private Set<SelectionKey> keys;
 | |
|   private Set<SelectionKey> selected;
 | |
| 
 | |
|   /**
 | |
|    * A dummy object whose monitor regulates access to both our
 | |
|    * selectThread and unhandledWakeup fields.
 | |
|    */
 | |
|   private Object selectThreadMutex = new Object ();
 | |
| 
 | |
|   /**
 | |
|    * Any thread that's currently blocked in a select operation.
 | |
|    */
 | |
|   private Thread selectThread;
 | |
| 
 | |
|   /**
 | |
|    * Indicates whether we have an unhandled wakeup call. This can
 | |
|    * be due to either wakeup() triggering a thread interruption while
 | |
|    * a thread was blocked in a select operation (in which case we need
 | |
|    * to reset this thread's interrupt status after interrupting the
 | |
|    * select), or else that no thread was on a select operation at the
 | |
|    * time that wakeup() was called, in which case the following select()
 | |
|    * operation should return immediately with nothing selected.
 | |
|    */
 | |
|   private boolean unhandledWakeup;
 | |
| 
 | |
|   public SelectorImpl (SelectorProvider provider)
 | |
|   {
 | |
|     super (provider);
 | |
| 
 | |
|     keys = new HashSet<SelectionKey> ();
 | |
|     selected = new HashSet<SelectionKey> ();
 | |
|   }
 | |
| 
 | |
|   protected void finalize() throws Throwable
 | |
|   {
 | |
|     close();
 | |
|   }
 | |
| 
 | |
|   protected final void implCloseSelector()
 | |
|     throws IOException
 | |
|   {
 | |
|     // Cancel any pending select operation.
 | |
|     wakeup();
 | |
| 
 | |
|     synchronized (keys)
 | |
|       {
 | |
|         synchronized (selected)
 | |
|           {
 | |
|             synchronized (cancelledKeys ())
 | |
|               {
 | |
|                 // FIXME: Release resources here.
 | |
|               }
 | |
|           }
 | |
|       }
 | |
|   }
 | |
| 
 | |
|   public final Set<SelectionKey> keys()
 | |
|   {
 | |
|     if (!isOpen())
 | |
|       throw new ClosedSelectorException();
 | |
| 
 | |
|     return Collections.unmodifiableSet (keys);
 | |
|   }
 | |
| 
 | |
|   public final int selectNow()
 | |
|     throws IOException
 | |
|   {
 | |
|     // FIXME: We're simulating an immediate select
 | |
|     // via a select with a timeout of one millisecond.
 | |
|     return select (1);
 | |
|   }
 | |
| 
 | |
|   public final int select()
 | |
|     throws IOException
 | |
|   {
 | |
|     return select (0);
 | |
|   }
 | |
| 
 | |
|   private final int[] getFDsAsArray (int ops)
 | |
|   {
 | |
|     int[] result;
 | |
|     int counter = 0;
 | |
|     Iterator<SelectionKey> it = keys.iterator ();
 | |
| 
 | |
|     // Count the number of file descriptors needed
 | |
|     while (it.hasNext ())
 | |
|       {
 | |
|         SelectionKeyImpl key = (SelectionKeyImpl) it.next ();
 | |
| 
 | |
|         if ((key.interestOps () & ops) != 0)
 | |
|           {
 | |
|             counter++;
 | |
|           }
 | |
|       }
 | |
| 
 | |
|     result = new int[counter];
 | |
| 
 | |
|     counter = 0;
 | |
|     it = keys.iterator ();
 | |
| 
 | |
|     // Fill the array with the file descriptors
 | |
|     while (it.hasNext ())
 | |
|       {
 | |
|         SelectionKeyImpl key = (SelectionKeyImpl) it.next ();
 | |
| 
 | |
|         if ((key.interestOps () & ops) != 0)
 | |
|           {
 | |
|             result[counter] = key.getNativeFD();
 | |
|             counter++;
 | |
|           }
 | |
|       }
 | |
| 
 | |
|     return result;
 | |
|   }
 | |
| 
 | |
|   public synchronized int select (long timeout)
 | |
|     throws IOException
 | |
|   {
 | |
|     if (!isOpen())
 | |
|       throw new ClosedSelectorException();
 | |
| 
 | |
|     synchronized (keys)
 | |
|       {
 | |
|         synchronized (selected)
 | |
|           {
 | |
|             deregisterCancelledKeys();
 | |
| 
 | |
|             // Set only keys with the needed interest ops into the arrays.
 | |
|             int[] read = getFDsAsArray (SelectionKey.OP_READ
 | |
|                                         | SelectionKey.OP_ACCEPT);
 | |
|             int[] write = getFDsAsArray (SelectionKey.OP_WRITE
 | |
|                                          | SelectionKey.OP_CONNECT);
 | |
| 
 | |
|             // FIXME: We dont need to check this yet
 | |
|             int[] except = new int [0];
 | |
| 
 | |
|             // Test to see if we've got an unhandled wakeup call,
 | |
|             // in which case we return immediately. Otherwise,
 | |
|             // remember our current thread and jump into the select.
 | |
|             // The monitor for dummy object selectThreadMutex regulates
 | |
|             // access to these fields.
 | |
| 
 | |
|             // FIXME: Not sure from the spec at what point we should
 | |
|             // return "immediately". Is it here or immediately upon
 | |
|             // entry to this function?
 | |
| 
 | |
|             // NOTE: There's a possibility of another thread calling
 | |
|             // wakeup() immediately after our thread releases
 | |
|             // selectThreadMutex's monitor here, in which case we'll
 | |
|             // do the select anyway. Since calls to wakeup() and select()
 | |
|             // among different threads happen in non-deterministic order,
 | |
|             // I don't think this is an issue.
 | |
|             synchronized (selectThreadMutex)
 | |
|               {
 | |
|                 if (unhandledWakeup)
 | |
|                   {
 | |
|                     unhandledWakeup = false;
 | |
|                     return 0;
 | |
|                   }
 | |
|                 else
 | |
|                   {
 | |
|                     selectThread = Thread.currentThread ();
 | |
|                   }
 | |
|               }
 | |
| 
 | |
|             // Call the native select() on all file descriptors.
 | |
|             int result = 0;
 | |
|             try
 | |
|               {
 | |
|                 begin();
 | |
|                 result = VMSelector.select (read, write, except, timeout);
 | |
|               }
 | |
|             finally
 | |
|               {
 | |
|                 end();
 | |
|               }
 | |
| 
 | |
|             // If our unhandled wakeup flag is set at this point,
 | |
|             // reset our thread's interrupt flag because we were
 | |
|             // awakened by wakeup() instead of an external thread
 | |
|             // interruption.
 | |
|             //
 | |
|             // NOTE: If we were blocked in a select() and one thread
 | |
|             // called Thread.interrupt() on the blocked thread followed
 | |
|             // by another thread calling Selector.wakeup(), then race
 | |
|             // conditions could make it so that the thread's interrupt
 | |
|             // flag is reset even though the Thread.interrupt() call
 | |
|             // "was there first". I don't think we need to care about
 | |
|             // this scenario.
 | |
|             synchronized (selectThreadMutex)
 | |
|               {
 | |
|                 if (unhandledWakeup)
 | |
|                   {
 | |
|                     unhandledWakeup = false;
 | |
|                     Thread.interrupted ();
 | |
|                   }
 | |
|                 selectThread = null;
 | |
|               }
 | |
| 
 | |
|             Iterator<SelectionKey> it = keys.iterator ();
 | |
| 
 | |
|             while (it.hasNext ())
 | |
|               {
 | |
|                 int ops = 0;
 | |
|                 SelectionKeyImpl key = (SelectionKeyImpl) it.next ();
 | |
| 
 | |
|                 // If key is already selected retrieve old ready ops.
 | |
|                 if (selected.contains (key))
 | |
|                   {
 | |
|                     ops = key.readyOps ();
 | |
|                   }
 | |
| 
 | |
|                 // Set new ready read/accept ops
 | |
|                 for (int i = 0; i < read.length; i++)
 | |
|                   {
 | |
|                     if (key.getNativeFD() == read[i])
 | |
|                       {
 | |
|                         if (key.channel () instanceof ServerSocketChannelImpl)
 | |
|                           {
 | |
|                             ops = ops | SelectionKey.OP_ACCEPT;
 | |
|                           }
 | |
|                         else
 | |
|                           {
 | |
|                             ops = ops | SelectionKey.OP_READ;
 | |
|                           }
 | |
|                       }
 | |
|                   }
 | |
| 
 | |
|                 // Set new ready write ops
 | |
|                 for (int i = 0; i < write.length; i++)
 | |
|                   {
 | |
|                     if (key.getNativeFD() == write[i])
 | |
|                       {
 | |
|                         if (key.channel() instanceof SocketChannel)
 | |
|                           {
 | |
|                             if (((SocketChannel) key.channel ()).isConnected ())
 | |
|                               ops = ops | SelectionKey.OP_WRITE;
 | |
|                             else
 | |
|                               ops = ops | SelectionKey.OP_CONNECT;
 | |
|                           }
 | |
|                         else
 | |
|                           ops = ops | SelectionKey.OP_WRITE;
 | |
|                       }
 | |
|                   }
 | |
| 
 | |
|                 // FIXME: We dont handle exceptional file descriptors yet.
 | |
| 
 | |
|                 // If key is not yet selected add it.
 | |
|                 if (!selected.contains (key))
 | |
|                   {
 | |
|                     selected.add (key);
 | |
|                   }
 | |
| 
 | |
|                 // Set new ready ops
 | |
|                 key.readyOps (key.interestOps () & ops);
 | |
|               }
 | |
|             deregisterCancelledKeys();
 | |
| 
 | |
|             return result;
 | |
|           }
 | |
|         }
 | |
|   }
 | |
| 
 | |
|   public final Set<SelectionKey> selectedKeys()
 | |
|   {
 | |
|     if (!isOpen())
 | |
|       throw new ClosedSelectorException();
 | |
| 
 | |
|     return selected;
 | |
|   }
 | |
| 
 | |
|   public final Selector wakeup()
 | |
|   {
 | |
|     // IMPLEMENTATION NOTE: Whereas the specification says that
 | |
|     // thread interruption should trigger a call to wakeup, we
 | |
|     // do the reverse under the covers: wakeup triggers a thread
 | |
|     // interrupt followed by a subsequent reset of the thread's
 | |
|     // interrupt status within select().
 | |
| 
 | |
|     // First, acquire the monitor of the object regulating
 | |
|     // access to our selectThread and unhandledWakeup fields.
 | |
|     synchronized (selectThreadMutex)
 | |
|       {
 | |
|         unhandledWakeup = true;
 | |
| 
 | |
|         // Interrupt any thread which is currently blocked in
 | |
|         // a select operation.
 | |
|         if (selectThread != null)
 | |
|           selectThread.interrupt ();
 | |
|       }
 | |
| 
 | |
|     return this;
 | |
|   }
 | |
| 
 | |
|   private final void deregisterCancelledKeys()
 | |
|   {
 | |
|     Set<SelectionKey> ckeys = cancelledKeys ();
 | |
|     synchronized (ckeys)
 | |
|     {
 | |
|       Iterator<SelectionKey> it = ckeys.iterator();
 | |
| 
 | |
|       while (it.hasNext ())
 | |
|         {
 | |
|           keys.remove ((SelectionKeyImpl) it.next ());
 | |
|           it.remove ();
 | |
|         }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   protected SelectionKey register (SelectableChannel ch, int ops, Object att)
 | |
|   {
 | |
|     return register ((AbstractSelectableChannel) ch, ops, att);
 | |
|   }
 | |
| 
 | |
|   protected final SelectionKey register (AbstractSelectableChannel ch, int ops,
 | |
|                                          Object att)
 | |
|   {
 | |
|     SelectionKeyImpl result;
 | |
| 
 | |
|     if (ch instanceof SocketChannelImpl)
 | |
|       result = new SocketChannelSelectionKey (ch, this);
 | |
|     else if (ch instanceof DatagramChannelImpl)
 | |
|       result = new DatagramChannelSelectionKey (ch, this);
 | |
|     else if (ch instanceof ServerSocketChannelImpl)
 | |
|       result = new ServerSocketChannelSelectionKey (ch, this);
 | |
|     else if (ch instanceof gnu.java.nio.SocketChannelImpl)
 | |
|       result = new gnu.java.nio.SocketChannelSelectionKeyImpl((gnu.java.nio.SocketChannelImpl)ch, this);
 | |
|     else
 | |
|       throw new InternalError ("No known channel type");
 | |
| 
 | |
|     synchronized (keys)
 | |
|       {
 | |
|         keys.add (result);
 | |
| 
 | |
|         result.interestOps (ops);
 | |
|         result.attach (att);
 | |
|       }
 | |
| 
 | |
|     return result;
 | |
|   }
 | |
| }
 |