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

org.jgroups.protocols.BARRIER Maven / Gradle / Ivy

Go to download

This artifact provides a single jar that contains all classes required to use remote Jakarta Enterprise Beans and Jakarta Messaging, including all dependencies. It is intended for use by those not using maven, maven users should just import the Jakarta Enterprise Beans and Jakarta Messaging BOM's instead (shaded JAR's cause lots of problems with maven, as it is very easy to inadvertently end up with different versions on classes on the class path).

There is a newer version: 35.0.0.Final
Show newest version
package org.jgroups.protocols;

import org.jgroups.Address;
import org.jgroups.Event;
import org.jgroups.Message;
import org.jgroups.TimeoutException;
import org.jgroups.annotations.MBean;
import org.jgroups.annotations.ManagedAttribute;
import org.jgroups.annotations.ManagedOperation;
import org.jgroups.annotations.Property;
import org.jgroups.stack.Protocol;
import org.jgroups.util.MessageBatch;
import org.jgroups.util.TimeScheduler;
import org.jgroups.util.Util;

import java.util.HashSet;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.Executor;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 * All messages up the stack have to go through a barrier (read lock, RL). By default, the barrier is open.
 * When a CLOSE_BARRIER event is received, we close the barrier by acquiring a write lock (WL). This succeeds when all
 * previous messages have completed (by releasing their RLs). Thus, when we have acquired the WL, we know that there
 * are no pending messages processed.
* When an OPEN_BARRIER event is received, we simply open the barrier again and let all messages pass in the up * direction. This is done by releasing the WL. * @author Bela Ban */ @MBean(description="Blocks all multicast threads when closed") public class BARRIER extends Protocol { @Property(description="Max time barrier can be closed. Default is 60000 ms") protected long max_close_time=60000; // how long can the barrier stay closed (in ms) ? 0 means forever @Property(description="Max time (in ms) to wait until the threads which passed the barrier before it was closed " + "have completed. If this time elapses, an exception will be thrown and state transfer will fail. 0 = wait forever") protected long flush_timeout=5000; protected final Lock lock=new ReentrantLock(); protected final AtomicBoolean barrier_closed=new AtomicBoolean(false); /** signals to waiting threads that the barrier is open again */ protected Condition no_pending_threads=lock.newCondition(); protected Map in_flight_threads=Util.createConcurrentMap(); protected volatile Future barrier_opener_future; protected TimeScheduler timer; protected Address local_addr; // mbrs from which unicasts should be accepted even if BARRIER is closed (PUNCH_HOLE adds, CLOSE_HOLE removes mbrs) protected final Set
holes=new HashSet<>(); // queues multicast messages or message batches (dest == null) protected final Map mcast_queue=new ConcurrentHashMap<>(); // queues unicast messages or message batches (dest != null) protected final Map ucast_queue=new ConcurrentHashMap<>(); protected TP transport; protected static final Object NULL=new Object(); @ManagedAttribute(description="Shows whether the barrier closed") public boolean isClosed() { return barrier_closed.get(); } @ManagedAttribute(description="Lists the members whose unicast messages are let through") public String getHoles() {return holes.toString();} public int getNumberOfInFlightThreads() { return in_flight_threads.size(); } @ManagedAttribute public int getInFlightThreadsCount() { return getNumberOfInFlightThreads(); } @ManagedAttribute public boolean isOpenerScheduled() { return barrier_opener_future != null && !barrier_opener_future.isDone() && !barrier_opener_future.isCancelled(); } public void init() throws Exception { super.init(); transport=getTransport(); timer=transport.getTimer(); } public void stop() { super.stop(); openBarrier(); } public void destroy() { super.destroy(); openBarrier(); } public Object down(Event evt) { switch(evt.getType()) { case Event.CLOSE_BARRIER: closeBarrier(); return null; case Event.OPEN_BARRIER: openBarrier(); return null; case Event.SET_LOCAL_ADDRESS: local_addr=(Address)evt.getArg(); break; case Event.PUNCH_HOLE: Address mbr=(Address)evt.getArg(); holes.add(mbr); return null; case Event.CLOSE_HOLE: mbr=(Address)evt.getArg(); holes.remove(mbr); return null; } return down_prot.down(evt); } public Object up(Event evt) { switch(evt.getType()) { case Event.MSG: Message msg=(Message)evt.getArg(); // https://issues.jboss.org/browse/JGRP-1341: let unicast messages pass if(msg.isFlagSet(Message.Flag.SKIP_BARRIER) || msg.getDest() != null && ((msg.isFlagSet(Message.Flag.OOB) && msg.isFlagSet(Message.Flag.INTERNAL)) || holes.contains(msg.getSrc()))) return up_prot.up(evt); if(barrier_closed.get()) { final Map map=msg.getDest() == null? mcast_queue : ucast_queue; map.put(msg.getSrc(), msg); return null; // queue and drop the message } Thread current_thread=Thread.currentThread(); in_flight_threads.put(current_thread, NULL); try { return up_prot.up(evt); } finally { unblock(current_thread); } case Event.CLOSE_BARRIER: closeBarrier(); return null; case Event.OPEN_BARRIER: openBarrier(); return null; } return up_prot.up(evt); } public void up(MessageBatch batch) { if(batch.dest() != null) { // let unicast message batches pass if((batch.mode() == MessageBatch.Mode.OOB && batch.mode() == MessageBatch.Mode.INTERNAL) || holes.contains(batch.sender())) { up_prot.up(batch); return; } } if(barrier_closed.get()) { final Map map=batch.dest() == null? mcast_queue : ucast_queue; map.put(batch.sender(), batch.last().putHeader(transport.getId(),new TpHeader(batch.clusterName()))); return; // queue the last message of the batch and drop the batch } Thread current_thread=Thread.currentThread(); in_flight_threads.put(current_thread, NULL); try { up_prot.up(batch); } finally { unblock(current_thread); } } protected void unblock(final Thread current_thread) { if(in_flight_threads.remove(current_thread) == NULL && in_flight_threads.isEmpty()) { lock.lock(); try { no_pending_threads.signalAll(); } finally { lock.unlock(); } } } /** Close the barrier. Temporarily remove all threads which are waiting or blocked, re-insert them after the call */ protected void closeBarrier() { if(!barrier_closed.compareAndSet(false, true)) return; // barrier is already closed long target_time=0, wait_time=0, start=System.currentTimeMillis(); in_flight_threads.remove(Thread.currentThread()); lock.lock(); try { // wait until all pending threads have returned while(barrier_closed.get() && !in_flight_threads.isEmpty()) { if(target_time == 0 && flush_timeout > 0) target_time=System.currentTimeMillis() + flush_timeout; for(Iterator it=in_flight_threads.keySet().iterator(); it.hasNext();) { Thread thread=it.next(); if(!thread.isAlive() || thread.getState() == Thread.State.TERMINATED) // should be the same it.remove(); } if(in_flight_threads.isEmpty()) break; try { if(flush_timeout <= 0) no_pending_threads.await(); else { if((wait_time=target_time - System.currentTimeMillis()) <= 0) break; no_pending_threads.await(wait_time, TimeUnit.MILLISECONDS); } } catch(InterruptedException e) { } } if(flush_timeout > 0 && !in_flight_threads.isEmpty()) { long time=System.currentTimeMillis() - start; throw new TimeoutException(local_addr + ": failed flushing pending threads in " + time + " ms; threads:\n" + printInFlightThreads()); } } finally { lock.unlock(); } if(max_close_time > 0) scheduleBarrierOpener(); } @ManagedOperation(description="Opens the barrier. No-op if already open") public void openBarrier() { if(!barrier_closed.compareAndSet(true, false)) return; // barrier was already open cancelBarrierOpener(); // cancels if running synchronized(mcast_queue) { flushQueue(mcast_queue); } synchronized(ucast_queue) { flushQueue(ucast_queue); } } @ManagedOperation(description="Lists the in-flight threads") protected String printInFlightThreads() { StringBuilder sb=new StringBuilder(); for(Thread thread: in_flight_threads.keySet()) sb.append(thread.toString()).append("\n"); return sb.toString(); } protected void flushQueue(final Map queue) { if(queue.isEmpty()) return; for(Message msg: queue.values()) { Executor pool=transport.pickThreadPool(msg.isFlagSet(Message.Flag.OOB),msg.isFlagSet(Message.Flag.INTERNAL)); try { pool.execute(transport.new SingleMessageHandler(msg)); } catch(Throwable t) { log.warn("%s: failure passing message up the stack: %s", local_addr, t); } } queue.clear(); } protected void scheduleBarrierOpener() { if(barrier_opener_future == null || barrier_opener_future.isDone()) { barrier_opener_future=timer.schedule(new Runnable() {public void run() {openBarrier();}}, max_close_time, TimeUnit.MILLISECONDS); } } protected void cancelBarrierOpener() { if(barrier_opener_future != null) { barrier_opener_future.cancel(true); barrier_opener_future=null; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy