/*
 * Copyright (c) 2008, 2011, Oracle and/or its affiliates. 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.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle 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 Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package sun.nio.fs;

import java.nio.file.*;
import java.util.*;

Base implementation class for watch keys.
/** * Base implementation class for watch keys. */
abstract class AbstractWatchKey implements WatchKey {
Maximum size of event list (in the future this may be tunable)
/** * Maximum size of event list (in the future this may be tunable) */
static final int MAX_EVENT_LIST_SIZE = 512;
Special event to signal overflow
/** * Special event to signal overflow */
static final Event<Object> OVERFLOW_EVENT = new Event<Object>(StandardWatchEventKinds.OVERFLOW, null);
Possible key states
/** * Possible key states */
private static enum State { READY, SIGNALLED }; // reference to watcher private final AbstractWatchService watcher; // reference to the original directory private final Path dir; // key state private State state; // pending events private List<WatchEvent<?>> events; // maps a context to the last event for the context (iff the last queued // event for the context is an ENTRY_MODIFY event). private Map<Object,WatchEvent<?>> lastModifyEvents; protected AbstractWatchKey(Path dir, AbstractWatchService watcher) { this.watcher = watcher; this.dir = dir; this.state = State.READY; this.events = new ArrayList<>(); this.lastModifyEvents = new HashMap<>(); } final AbstractWatchService watcher() { return watcher; }
Return the original watchable (Path)
/** * Return the original watchable (Path) */
@Override public Path watchable() { return dir; }
Enqueues this key to the watch service
/** * Enqueues this key to the watch service */
final void signal() { synchronized (this) { if (state == State.READY) { state = State.SIGNALLED; watcher.enqueueKey(this); } } }
Adds the event to this key and signals it.
/** * Adds the event to this key and signals it. */
@SuppressWarnings("unchecked") final void signalEvent(WatchEvent.Kind<?> kind, Object context) { boolean isModify = (kind == StandardWatchEventKinds.ENTRY_MODIFY); synchronized (this) { int size = events.size(); if (size > 0) { // if the previous event is an OVERFLOW event or this is a // repeated event then we simply increment the counter WatchEvent<?> prev = events.get(size-1); if ((prev.kind() == StandardWatchEventKinds.OVERFLOW) || ((kind == prev.kind() && Objects.equals(context, prev.context())))) { ((Event<?>)prev).increment(); return; } // if this is a modify event and the last entry for the context // is a modify event then we simply increment the count if (!lastModifyEvents.isEmpty()) { if (isModify) { WatchEvent<?> ev = lastModifyEvents.get(context); if (ev != null) { assert ev.kind() == StandardWatchEventKinds.ENTRY_MODIFY; ((Event<?>)ev).increment(); return; } } else { // not a modify event so remove from the map as the // last event will no longer be a modify event. lastModifyEvents.remove(context); } } // if the list has reached the limit then drop pending events // and queue an OVERFLOW event if (size >= MAX_EVENT_LIST_SIZE) { kind = StandardWatchEventKinds.OVERFLOW; isModify = false; context = null; } } // non-repeated event Event<Object> ev = new Event<>((WatchEvent.Kind<Object>)kind, context); if (isModify) { lastModifyEvents.put(context, ev); } else if (kind == StandardWatchEventKinds.OVERFLOW) { // drop all pending events events.clear(); lastModifyEvents.clear(); } events.add(ev); signal(); } } @Override public final List<WatchEvent<?>> pollEvents() { synchronized (this) { List<WatchEvent<?>> result = events; events = new ArrayList<>(); lastModifyEvents.clear(); return result; } } @Override public final boolean reset() { synchronized (this) { if (state == State.SIGNALLED && isValid()) { if (events.isEmpty()) { state = State.READY; } else { // pending events so re-queue key watcher.enqueueKey(this); } } return isValid(); } }
WatchEvent implementation
/** * WatchEvent implementation */
private static class Event<T> implements WatchEvent<T> { private final WatchEvent.Kind<T> kind; private final T context; // synchronize on watch key to access/increment count private int count; Event(WatchEvent.Kind<T> type, T context) { this.kind = type; this.context = context; this.count = 1; } @Override public WatchEvent.Kind<T> kind() { return kind; } @Override public T context() { return context; } @Override public int count() { return count; } // for repeated events void increment() { count++; } } }