/* * The MIT License * * Copyright (c) 2010, InfraDNA, Inc. * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ package hudson.model.queue; import hudson.AbortException; /** * A concurrency primitive that waits for N number of threads to synchronize, * then have them all execute a certain task, then synchronize for the completion of all N tasks. * * If any of the threads are interrupted while waiting for the completion of the condition or * fails to complete the task, then all the involved threads get interrupted and receive a failure. * * @author Kohsuke Kawaguchi */ class Latch { private final int n; private int i=0; /** * If the synchronization on the latch is aborted/interrupted, * point to the stack trace where that happened. If null, * no interruption happened. */ private Exception interrupted; /** * @param n * Number of threads that need to enter the {@link #synchronize()} method * before all of them get released. */ public Latch(int n) { this.n = n; } public synchronized void abort(Throwable cause) { interrupted = new AbortException(); if (cause!=null) interrupted.initCause(cause); notifyAll(); } /** * Waits for N threads to enter the {@link #synchronize()} method, then * returns. * * @return * returns normally if N threads successfully synchronized. * @throws InterruptedException * if any of the threads that were synchronizing get interrupted, * or if the {@link #abort(Throwable)} is called. */ public synchronized void synchronize() throws InterruptedException { check(n); try { onCriteriaMet(); } catch (Error e) { abort(e); throw e; } catch (RuntimeException e) { abort(e); throw e; } check(n*2); } private void check(int threshold) throws InterruptedException { i++; if (i==threshold) { notifyAll(); } else { while (i<threshold && interrupted==null) { try { wait(); } catch (InterruptedException e) { interrupted = e; notifyAll(); throw e; } } } // all of us either leave normally or get interrupted if (interrupted!=null) throw (InterruptedException)new InterruptedException().initCause(interrupted); } /** * Once N threads gather, this method is called to execute to carry out some task. */ protected void onCriteriaMet() throws InterruptedException {} }