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

zmq.YPipe Maven / Gradle / Ivy

There is a newer version: 0.6.0
Show newest version
/*  
    Copyright (c) 2009-2011 250bpm s.r.o.
    Copyright (c) 2007-2009 iMatix Corporation
    Copyright (c) 2007-2011 Other contributors as noted in the AUTHORS file

    This file is part of 0MQ.

    0MQ 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 3 of the License, or
    (at your option) any later version.

    0MQ 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 program.  If not, see .
*/
package zmq;

import java.util.concurrent.atomic.AtomicInteger;


public class YPipe {

    //  Allocation-efficient queue to store pipe items.
    //  Front of the queue points to the first prefetched item, back of
    //  the pipe points to last un-flushed item. Front is used only by
    //  reader thread, while back is used only by writer thread.
    private final YQueue queue;

    //  Points to the first un-flushed item. This variable is used
    //  exclusively by writer thread.
    private int w;

    //  Points to the first un-prefetched item. This variable is used
    //  exclusively by reader thread.
    private int r;

    //  Points to the first item to be flushed in the future.
    private int f;

    //  The single point of contention between writer and reader thread.
    //  Points past the last flushed item. If it is NULL,
    //  reader is asleep. This pointer should be always accessed using
    //  atomic operations.
    private final AtomicInteger c;

    public YPipe(int qsize) {
        queue = new YQueue(qsize);
        w = r = f = queue.back_pos();
        c = new AtomicInteger(queue.back_pos());
            
    }

    //  Write an item to the pipe.  Don't flush it yet. If incomplete is
    //  set to true the item is assumed to be continued by items
    //  subsequently written to the pipe. Incomplete items are never
    //  flushed down the stream.
    public final void write (final T value_, boolean incomplete_)
    {
        //  Place the value to the queue, add new terminator element.
        queue.push(value_);

        //  Move the "flush up to here" poiter.
        if (!incomplete_) {
            f = queue.back_pos();
        }
    }
    
    //  Pop an incomplete item from the pipe. Returns true is such
    //  item exists, false otherwise.
    public T unwrite ()
    {
        
        if (f == queue.back_pos())
            return null;
        queue.unpush();
        return queue.back();
    }
    
    //  Flush all the completed items into the pipe. Returns false if
    //  the reader thread is sleeping. In that case, caller is obliged to
    //  wake the reader up before using the pipe again.
    public final boolean flush ()
    {
        //  If there are no un-flushed items, do nothing.
        if (w == f) {
            return true;
        }

        
        //  Try to set 'c' to 'f'.
        if (!c.compareAndSet(w, f)) {

            //  Compare-and-swap was unseccessful because 'c' is NULL.
            //  This means that the reader is asleep. Therefore we don't
            //  care about thread-safeness and update c in non-atomic
            //  manner. We'll return false to let the caller know
            //  that reader is sleeping.
            c.set (f);
            w = f;
            return false;
        }
        
        //  Reader is alive. Nothing special to do now. Just move
        //  the 'first un-flushed item' pointer to 'f'.
        w = f;
        return true;
    }
    
    //  Check whether item is available for reading.
    public final boolean check_read ()
    {
        //  Was the value prefetched already? If so, return.
        int h = queue.front_pos();
        if (h != r) 
             return true;

        //  There's no prefetched value, so let us prefetch more values.
        //  Prefetching is to simply retrieve the
        //  pointer from c in atomic fashion. If there are no
        //  items to prefetch, set c to -1 (using compare-and-swap).
        if (c.compareAndSet (h, -1)) {
             // nothing to read, h == r must be the same
        } else {
            // something to have been written
             r = c.get();
        }
        
        //  If there are no elements prefetched, exit.
        //  During pipe's lifetime r should never be NULL, however,
        //  it can happen during pipe shutdown when items
        //  are being deallocated.
        if (h == r || r == -1) 
            return false;

        //  There was at least one value prefetched.
        return true;
    }


    //  Reads an item from the pipe. Returns false if there is no value.
    //  available.
    public final T read ()
    {
        //  Try to prefetch a value.
        if (!check_read ())
            return null;

        //  There was at least one value prefetched.
        //  Return it to the caller.
        T value_ = queue.pop();
        
        return value_;
    }
    
    
    //  Applies the function fn to the first elemenent in the pipe
    //  and returns the value returned by the fn.
    //  The pipe mustn't be empty or the function crashes.
    public final T probe() {
        
        boolean rc = check_read ();
        assert (rc);
        
        T value = queue.front ();
        return value;
    }


}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy