You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

Parallel.java 6.7 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. /*
  2. * The Apache Software License, Version 1.1
  3. *
  4. * Copyright (c) 2001-2002 The Apache Software Foundation. All rights
  5. * reserved.
  6. *
  7. * Redistribution and use in source and binary forms, with or without
  8. * modification, are permitted provided that the following conditions
  9. * are met:
  10. *
  11. * 1. Redistributions of source code must retain the above copyright
  12. * notice, this list of conditions and the following disclaimer.
  13. *
  14. * 2. Redistributions in binary form must reproduce the above copyright
  15. * notice, this list of conditions and the following disclaimer in
  16. * the documentation and/or other materials provided with the
  17. * distribution.
  18. *
  19. * 3. The end-user documentation included with the redistribution, if
  20. * any, must include the following acknowlegement:
  21. * "This product includes software developed by the
  22. * Apache Software Foundation (http://www.apache.org/)."
  23. * Alternately, this acknowlegement may appear in the software itself,
  24. * if and wherever such third-party acknowlegements normally appear.
  25. *
  26. * 4. The names "The Jakarta Project", "Ant", and "Apache Software
  27. * Foundation" must not be used to endorse or promote products derived
  28. * from this software without prior written permission. For written
  29. * permission, please contact apache@apache.org.
  30. *
  31. * 5. Products derived from this software may not be called "Apache"
  32. * nor may "Apache" appear in their names without prior written
  33. * permission of the Apache Group.
  34. *
  35. * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
  36. * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
  37. * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  38. * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
  39. * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  40. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  41. * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
  42. * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  43. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  44. * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
  45. * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  46. * SUCH DAMAGE.
  47. * ====================================================================
  48. *
  49. * This software consists of voluntary contributions made by many
  50. * individuals on behalf of the Apache Software Foundation. For more
  51. * information on the Apache Software Foundation, please see
  52. * <http://www.apache.org/>.
  53. */
  54. package org.apache.tools.ant.taskdefs;
  55. import org.apache.tools.ant.Task;
  56. import org.apache.tools.ant.TaskContainer;
  57. import org.apache.tools.ant.BuildException;
  58. import org.apache.tools.ant.Location;
  59. import java.util.Vector;
  60. import java.util.Enumeration;
  61. /**
  62. * Implements a multi threaded task execution.
  63. * <p>
  64. * @author Thomas Christen <a href="mailto:chr@active.ch">chr@active.ch</a>
  65. * @author <a href="mailto:conor@apache.org">Conor MacNeill </a>
  66. *
  67. * @ant.task category="control"
  68. */
  69. public class Parallel extends Task
  70. implements TaskContainer {
  71. /** Collection holding the nested tasks */
  72. private Vector nestedTasks = new Vector();
  73. /**
  74. * Add a nested task to execute parallel (asynchron).
  75. * <p>
  76. * @param nestedTask Nested task to be executed in parallel
  77. */
  78. public void addTask(Task nestedTask) throws BuildException {
  79. nestedTasks.addElement(nestedTask);
  80. }
  81. /**
  82. * Block execution until the specified time or for a
  83. * specified amount of milliseconds and if defined,
  84. * execute the wait status.
  85. */
  86. public void execute() throws BuildException {
  87. TaskThread[] threads = new TaskThread[nestedTasks.size()];
  88. int threadNumber = 0;
  89. for (Enumeration e = nestedTasks.elements(); e.hasMoreElements(); threadNumber++) {
  90. Task nestedTask = (Task)e.nextElement();
  91. threads[threadNumber] = new TaskThread(threadNumber, nestedTask);
  92. }
  93. // now start all threads
  94. for (int i = 0; i < threads.length; ++i) {
  95. threads[i].start();
  96. }
  97. // now join to all the threads
  98. for (int i = 0; i < threads.length; ++i) {
  99. try {
  100. threads[i].join();
  101. }
  102. catch (InterruptedException ie) {
  103. // who would interrupt me at a time like this?
  104. }
  105. }
  106. // now did any of the threads throw an exception
  107. StringBuffer exceptionMessage = new StringBuffer();
  108. String lSep = System.getProperty("line.separator");
  109. int numExceptions = 0;
  110. Throwable firstException = null;
  111. Location firstLocation = Location.UNKNOWN_LOCATION;;
  112. for (int i = 0; i < threads.length; ++i) {
  113. Throwable t = threads[i].getException();
  114. if (t != null) {
  115. numExceptions++;
  116. if (firstException == null) {
  117. firstException = t;
  118. }
  119. if (t instanceof BuildException &&
  120. firstLocation == Location.UNKNOWN_LOCATION) {
  121. firstLocation = ((BuildException)t).getLocation();
  122. }
  123. exceptionMessage.append(lSep);
  124. exceptionMessage.append(t.getMessage());
  125. }
  126. }
  127. if (numExceptions == 1) {
  128. if (firstException instanceof BuildException) {
  129. throw (BuildException)firstException;
  130. }
  131. else {
  132. throw new BuildException(firstException);
  133. }
  134. }
  135. else if (numExceptions > 1) {
  136. throw new BuildException(exceptionMessage.toString(), firstLocation);
  137. }
  138. }
  139. class TaskThread extends Thread {
  140. private Throwable exception;
  141. private Task task;
  142. private int taskNumber;
  143. /**
  144. * Construct a new TaskThread.<p>
  145. *
  146. * @param task the Task to be executed in a seperate thread
  147. */
  148. TaskThread(int taskNumber, Task task) {
  149. this.task = task;
  150. this.taskNumber = taskNumber;
  151. }
  152. /**
  153. * Executes the task within a thread and takes care about
  154. * Exceptions raised within the task.
  155. */
  156. public void run() {
  157. try {
  158. task.perform();
  159. }
  160. catch (Throwable t) {
  161. exception = t;
  162. }
  163. }
  164. public Throwable getException() {
  165. return exception;
  166. }
  167. }
  168. }