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.9 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190
  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 java.util.Enumeration;
  56. import java.util.Vector;
  57. import org.apache.tools.ant.BuildException;
  58. import org.apache.tools.ant.Location;
  59. import org.apache.tools.ant.Task;
  60. import org.apache.tools.ant.TaskContainer;
  61. import org.apache.tools.ant.util.StringUtils;
  62. /**
  63. * Executes the contained tasks in separate threads, continuing
  64. * once all are completed.
  65. * <p>
  66. * @author Thomas Christen <a href="mailto:chr@active.ch">chr@active.ch</a>
  67. * @author Conor MacNeill
  68. * @since Ant 1.4
  69. *
  70. * @ant.task category="control"
  71. */
  72. public class Parallel extends Task
  73. implements TaskContainer {
  74. /** Collection holding the nested tasks */
  75. private Vector nestedTasks = new Vector();
  76. /**
  77. * Add a nested task to execute in parallel.
  78. * @param nestedTask Nested task to be executed in parallel
  79. */
  80. public void addTask(Task nestedTask) throws BuildException {
  81. nestedTasks.addElement(nestedTask);
  82. }
  83. /**
  84. * Block execution until the specified time or for a
  85. * specified amount of milliseconds and if defined,
  86. * execute the wait status.
  87. */
  88. public void execute() throws BuildException {
  89. TaskThread[] threads = new TaskThread[nestedTasks.size()];
  90. int threadNumber = 0;
  91. for (Enumeration e = nestedTasks.elements(); e.hasMoreElements();
  92. threadNumber++) {
  93. Task nestedTask = (Task) e.nextElement();
  94. threads[threadNumber] = new TaskThread(threadNumber, nestedTask);
  95. }
  96. // now start all threads
  97. for (int i = 0; i < threads.length; ++i) {
  98. threads[i].start();
  99. }
  100. // now join to all the threads
  101. for (int i = 0; i < threads.length; ++i) {
  102. try {
  103. threads[i].join();
  104. } catch (InterruptedException ie) {
  105. // who would interrupt me at a time like this?
  106. }
  107. }
  108. // now did any of the threads throw an exception
  109. StringBuffer exceptionMessage = new StringBuffer();
  110. int numExceptions = 0;
  111. Throwable firstException = null;
  112. Location firstLocation = Location.UNKNOWN_LOCATION;;
  113. for (int i = 0; i < threads.length; ++i) {
  114. Throwable t = threads[i].getException();
  115. if (t != null) {
  116. numExceptions++;
  117. if (firstException == null) {
  118. firstException = t;
  119. }
  120. if (t instanceof BuildException &&
  121. firstLocation == Location.UNKNOWN_LOCATION) {
  122. firstLocation = ((BuildException) t).getLocation();
  123. }
  124. exceptionMessage.append(StringUtils.LINE_SEP);
  125. exceptionMessage.append(t.getMessage());
  126. }
  127. }
  128. if (numExceptions == 1) {
  129. if (firstException instanceof BuildException) {
  130. throw (BuildException) firstException;
  131. } else {
  132. throw new BuildException(firstException);
  133. }
  134. } else if (numExceptions > 1) {
  135. throw new BuildException(exceptionMessage.toString(),
  136. firstLocation);
  137. }
  138. }
  139. /**
  140. * thread that execs a task
  141. */
  142. class TaskThread extends Thread {
  143. private Throwable exception;
  144. private Task task;
  145. private int taskNumber;
  146. /**
  147. * Construct a new TaskThread.<p>
  148. *
  149. * @param task the Task to be executed in a seperate thread
  150. */
  151. TaskThread(int taskNumber, Task task) {
  152. this.task = task;
  153. this.taskNumber = taskNumber;
  154. }
  155. /**
  156. * Executes the task within a thread and takes care about
  157. * Exceptions raised within the task.
  158. */
  159. public void run() {
  160. try {
  161. task.perform();
  162. } catch (Throwable t) {
  163. exception = t;
  164. }
  165. }
  166. /**
  167. * get any exception that got thrown during execution;
  168. * @return an exception or null for no exception/not yet finished
  169. */
  170. public Throwable getException() {
  171. return exception;
  172. }
  173. }
  174. }