All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.threerings.presents.util.FutureResult Maven / Gradle / Ivy

//
// $Id: FutureResult.java 6407 2011-01-01 05:02:21Z dhoover $
//
// Narya library - tools for developing networked games
// Copyright (C) 2002-2011 Three Rings Design, Inc., All Rights Reserved
// http://code.google.com/p/narya/
//
// This library is free software; 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 2.1 of the License, or
// (at your option) any later version.
//
// This library 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 library; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA

package com.threerings.presents.util;

import java.util.concurrent.CancellationException;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.concurrent.FutureTask;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.locks.AbstractQueuedSynchronizer;

import com.threerings.presents.client.InvocationService;
import com.threerings.presents.server.InvocationException;

/**
 * Provides the result of an invocation service request as a future result. Modeled on and mostly
 * implemented by code from {@link FutureTask}.
 */
public class FutureResult
    implements Future, InvocationService.ResultListener
{
    // from interface Future
    public boolean cancel (boolean mayInterruptIfRunning)
    {
        return _sync.innerCancel(mayInterruptIfRunning);
    }

    // from interface Future
    public boolean isCancelled ()
    {
        return _sync.innerIsCancelled();
    }

    // from interface Future
    public boolean isDone ()
    {
        return _sync.innerIsDone();
    }

    // from interface Future
    public V get () throws InterruptedException, ExecutionException
    {
        return _sync.innerGet();
    }

    // from interface Future
    public V get (long timeout, TimeUnit unit)
        throws InterruptedException, ExecutionException, TimeoutException
    {
        return _sync.innerGet(unit.toNanos(timeout));
    }

    // from interface InvocationService.ResultListener
    public void requestProcessed (Object result)
    {
        @SuppressWarnings("unchecked") V value = (V)result;
        _sync.innerSet(value);
    }

    // from interface InvocationService.ResultListener
    public void requestFailed (String cause)
    {
        _sync.innerSetException(new InvocationException(cause));
    }

    protected class Sync extends AbstractQueuedSynchronizer
    {
        public boolean innerIsCancelled () {
            return getState() == CANCELLED;
        }

        public boolean innerIsDone () {
            return ranOrCancelled(getState());
        }

        public V innerGet() throws InterruptedException, ExecutionException {
            acquireSharedInterruptibly(0);
            if (getState() == CANCELLED) {
                throw new CancellationException();
            }
            if (_exception != null) {
                throw new ExecutionException(_exception);
            }
            return _result;
        }

        public V innerGet (long nanosTimeout)
            throws InterruptedException, ExecutionException, TimeoutException {
            if (!tryAcquireSharedNanos(0, nanosTimeout)) {
                throw new TimeoutException();
            }
            if (getState() == CANCELLED) {
                throw new CancellationException();
            }
            if (_exception != null) {
                throw new ExecutionException(_exception);
            }
            return _result;
        }

        public void innerSet (V v) {
            for (;;) {
                int s = getState();
                if (s == RAN) {
                    return;
                }
                if (s == CANCELLED) {
                    releaseShared(0);
                    return;
                }
                if (compareAndSetState(s, RAN)) {
                    _result = v;
                    releaseShared(0);
                    return;
                }
            }
        }

        public void innerSetException (Throwable t) {
            for (;;) {
                int s = getState();
                if (s == RAN) {
                    return;
                }
                if (s == CANCELLED) {
                    releaseShared(0);
                    return;
                }
                if (compareAndSetState(s, RAN)) {
                    _exception = t;
                    _result = null;
                    releaseShared(0);
                    return;
                }
            }
        }

        public boolean innerCancel (boolean mayInterruptIfRunning) {
            for (;;) {
                int s = getState();
                if (ranOrCancelled(s)) {
                    return false;
                }
                if (compareAndSetState(s, CANCELLED)) {
                    break;
                }
            }
            releaseShared(0);
            return true;
        }

        @Override protected int tryAcquireShared (int ignore) {
            return innerIsDone()? 1 : -1;
        }

        @Override protected boolean tryReleaseShared (int ignore) {
            return true;
        }

        protected boolean ranOrCancelled (int state) {
            return (state & (RAN | CANCELLED)) != 0;
        }

        /** The result to return from get() */
        protected V _result;

        /** The exception to throw from get() */
        protected Throwable _exception;

        protected static final int RUNNING = 1;
        protected static final int RAN = 2;
        protected static final int CANCELLED = 4;
    }

    protected Sync _sync = new Sync();
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy