/* * Copyright 2008-2009 Sun Microsystems, Inc. All Rights Reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Sun designates this * particular file as subject to the "Classpath" exception as provided * by Sun in the LICENSE file that accompanied this code. * * 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 General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, * CA 95054 USA or visit www.sun.com if you need additional information or * have any questions. */ package com.barbarysoftware.watchservice; import java.io.IOException; import java.util.concurrent.LinkedBlockingDeque; import java.util.concurrent.TimeUnit; /** * Base implementation class for watch services. */ abstract class AbstractWatchService extends WatchService { // signaled keys waiting to be dequeued private final LinkedBlockingDeque<WatchKey> pendingKeys = new LinkedBlockingDeque<WatchKey>(); // special key to indicate that watch service is closed private final WatchKey CLOSE_KEY = new AbstractWatchKey(null) { @Override public boolean isValid() { return true; } @Override public void cancel() { } }; // used when closing watch service private volatile boolean closed; private final Object closeLock = new Object(); protected AbstractWatchService() { } /** * Register the given object with this watch service */ abstract WatchKey register(WatchableFile watchableFile, WatchEvent.Kind<?>[] events, WatchEvent.Modifier... modifers) throws IOException; // used by AbstractWatchKey to enqueue key final void enqueueKey(WatchKey key) { pendingKeys.offer(key); } /** * Throws ClosedWatchServiceException if watch service is closed */ private void checkOpen() { if (closed) throw new ClosedWatchServiceException(); } /** * Checks the key isn't the special CLOSE_KEY used to unblock threads when * the watch service is closed. */ private void checkKey(WatchKey key) { if (key == CLOSE_KEY) { // re-queue in case there are other threads blocked in take/poll enqueueKey(key); } checkOpen(); } @Override public final WatchKey poll() { checkOpen(); WatchKey key = pendingKeys.poll(); checkKey(key); return key; } @Override public final WatchKey poll(long timeout, TimeUnit unit) throws InterruptedException { checkOpen(); WatchKey key = pendingKeys.poll(timeout, unit); checkKey(key); return key; } @Override public final WatchKey take() throws InterruptedException { checkOpen(); WatchKey key = pendingKeys.take(); checkKey(key); return key; } /** * Tells whether or not this watch service is open. */ final boolean isOpen() { return !closed; } /** * Retrieves the object upon which the close method synchronizes. */ final Object closeLock() { return closeLock; } /** * Closes this watch service. This method is invoked by the close * method to perform the actual work of closing the watch service. */ abstract void implClose() throws IOException; @Override public final void close() throws IOException { synchronized (closeLock) { // nothing to do if already closed if (closed) return; closed = true; implClose(); // clear pending keys and queue special key to ensure that any // threads blocked in take/poll wakeup pendingKeys.clear(); pendingKeys.offer(CLOSE_KEY); } } }