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

StreamCopyThread.java 4.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. /*
  2. * Copyright (C) 2009-2010, Google Inc.
  3. * and other copyright owners as documented in the project's IP log.
  4. *
  5. * This program and the accompanying materials are made available
  6. * under the terms of the Eclipse Distribution License v1.0 which
  7. * accompanies this distribution, is reproduced below, and is
  8. * available at http://www.eclipse.org/org/documents/edl-v10.php
  9. *
  10. * All rights reserved.
  11. *
  12. * Redistribution and use in source and binary forms, with or
  13. * without modification, are permitted provided that the following
  14. * conditions are met:
  15. *
  16. * - Redistributions of source code must retain the above copyright
  17. * notice, this list of conditions and the following disclaimer.
  18. *
  19. * - Redistributions in binary form must reproduce the above
  20. * copyright notice, this list of conditions and the following
  21. * disclaimer in the documentation and/or other materials provided
  22. * with the distribution.
  23. *
  24. * - Neither the name of the Eclipse Foundation, Inc. nor the
  25. * names of its contributors may be used to endorse or promote
  26. * products derived from this software without specific prior
  27. * written permission.
  28. *
  29. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
  30. * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
  31. * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
  32. * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  33. * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
  34. * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  35. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
  36. * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  37. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  38. * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
  39. * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  40. * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
  41. * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  42. */
  43. package org.eclipse.jgit.util.io;
  44. import java.io.IOException;
  45. import java.io.InputStream;
  46. import java.io.InterruptedIOException;
  47. import java.io.OutputStream;
  48. import java.util.concurrent.atomic.AtomicInteger;
  49. /** Thread to copy from an input stream to an output stream. */
  50. public class StreamCopyThread extends Thread {
  51. private static final int BUFFER_SIZE = 1024;
  52. private final InputStream src;
  53. private final OutputStream dst;
  54. private final AtomicInteger flushCounter = new AtomicInteger(0);
  55. private volatile boolean done;
  56. /**
  57. * Create a thread to copy data from an input stream to an output stream.
  58. *
  59. * @param i
  60. * stream to copy from. The thread terminates when this stream
  61. * reaches EOF. The thread closes this stream before it exits.
  62. * @param o
  63. * stream to copy into. The destination stream is automatically
  64. * closed when the thread terminates.
  65. */
  66. public StreamCopyThread(final InputStream i, final OutputStream o) {
  67. setName(Thread.currentThread().getName() + "-StreamCopy");
  68. src = i;
  69. dst = o;
  70. }
  71. /**
  72. * Request the thread to flush the output stream as soon as possible.
  73. * <p>
  74. * This is an asynchronous request to the thread. The actual flush will
  75. * happen at some future point in time, when the thread wakes up to process
  76. * the request.
  77. */
  78. public void flush() {
  79. flushCounter.incrementAndGet();
  80. interrupt();
  81. }
  82. /**
  83. * Request that the thread terminate, and wait for it.
  84. * <p>
  85. * This method signals to the copy thread that it should stop as soon as
  86. * there is no more IO occurring.
  87. *
  88. * @throws InterruptedException
  89. * the calling thread was interrupted.
  90. */
  91. public void halt() throws InterruptedException {
  92. for (;;) {
  93. join(250 /* milliseconds */);
  94. if (isAlive()) {
  95. done = true;
  96. interrupt();
  97. } else
  98. break;
  99. }
  100. }
  101. @Override
  102. public void run() {
  103. try {
  104. final byte[] buf = new byte[BUFFER_SIZE];
  105. for (;;) {
  106. try {
  107. if (needFlush())
  108. dst.flush();
  109. if (done)
  110. break;
  111. final int n;
  112. try {
  113. n = src.read(buf);
  114. } catch (InterruptedIOException wakey) {
  115. continue;
  116. }
  117. if (n < 0)
  118. break;
  119. for (;;) {
  120. try {
  121. dst.write(buf, 0, n);
  122. } catch (InterruptedIOException wakey) {
  123. continue;
  124. }
  125. break;
  126. }
  127. } catch (IOException e) {
  128. break;
  129. }
  130. }
  131. } finally {
  132. try {
  133. src.close();
  134. } catch (IOException e) {
  135. // Ignore IO errors on close
  136. }
  137. try {
  138. dst.close();
  139. } catch (IOException e) {
  140. // Ignore IO errors on close
  141. }
  142. }
  143. }
  144. private boolean needFlush() {
  145. int i = flushCounter.get();
  146. if (i > 0) {
  147. flushCounter.decrementAndGet();
  148. return true;
  149. }
  150. return false;
  151. }
  152. }