/amibe/src/org/jcae/mesh/xmldata/DoubleFileReaderByDirectBuffer.java

http://github.com/jeromerobert/jCAE · Java · 225 lines · 171 code · 26 blank · 28 comment · 29 complexity · 114e0e8d7f8af784979f2b67c8229180 MD5 · raw file

  1. /*
  2. * Project Info: http://jcae.sourceforge.net
  3. *
  4. * This program is free software; you can redistribute it and/or modify it under
  5. * the terms of the GNU Lesser General Public License as published by the Free
  6. * Software Foundation; either version 2.1 of the License, or (at your option)
  7. * any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful, but WITHOUT
  10. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
  11. * FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
  12. * details.
  13. *
  14. * You should have received a copy of the GNU Lesser General Public License
  15. * along with this program; if not, write to the Free Software Foundation, Inc.,
  16. * 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA.
  17. *
  18. * (C) Copyright 2009, by EADS France
  19. */
  20. package org.jcae.mesh.xmldata;
  21. import java.io.File;
  22. import java.io.FileInputStream;
  23. import java.io.FileNotFoundException;
  24. import java.io.IOException;
  25. import java.nio.ByteBuffer;
  26. import java.nio.Buffer;
  27. import java.nio.DoubleBuffer;
  28. import java.nio.channels.FileChannel;
  29. public class DoubleFileReaderByDirectBuffer implements DoubleFileReader
  30. {
  31. private final static int BUFFER_SIZE = 8*1024*1024;
  32. private final static int ARRAY_SIZE = 4096;
  33. private static final int ELEMENT_SIZE = 8;
  34. private final FileChannel fc;
  35. private final long numberOfElements;
  36. private final ByteBuffer bb;
  37. private final DoubleBuffer tb;
  38. private double[] array;
  39. private int startBufferIndex;
  40. private int arrayIndex;
  41. private int remaining;
  42. public DoubleFileReaderByDirectBuffer(File file) throws IOException
  43. {
  44. fc = new FileInputStream(file).getChannel();
  45. numberOfElements = fc.size() / ELEMENT_SIZE;
  46. if (fc.size() > BUFFER_SIZE)
  47. bb = ByteBuffer.allocateDirect(BUFFER_SIZE);
  48. else
  49. bb = ByteBuffer.allocateDirect((int) fc.size());
  50. tb = bb.asDoubleBuffer();
  51. // Copy from file into buffer
  52. copyFileIntoBuffer();
  53. }
  54. public long size()
  55. {
  56. return numberOfElements;
  57. }
  58. private void moveBufferTo(long newPosition) throws IOException
  59. {
  60. fc.position(newPosition);
  61. // Fill in buffer
  62. copyFileIntoBuffer();
  63. }
  64. public double get() throws IOException
  65. {
  66. if (remaining == 0)
  67. copyBufferIntoArray();
  68. if (remaining < 0)
  69. throw new IOException();
  70. double ret = array[arrayIndex];
  71. arrayIndex++;
  72. remaining--;
  73. // We must check for EOF if all values are exhausted
  74. if (remaining == 0 && !tb.hasRemaining())
  75. copyFileIntoBuffer();
  76. return ret;
  77. }
  78. public double get(int index) throws IOException
  79. {
  80. int relIndex = index - startBufferIndex;
  81. if (relIndex < 0 || relIndex >= tb.limit())
  82. {
  83. moveBufferTo(index * ELEMENT_SIZE);
  84. if (remaining < 0)
  85. throw new IndexOutOfBoundsException();
  86. relIndex = index - startBufferIndex;
  87. }
  88. return tb.get(relIndex);
  89. }
  90. public int get(double[] dst) throws IOException
  91. {
  92. return get(dst, 0, dst.length);
  93. }
  94. public final int get(double[] dst, int pos, int len) throws IOException
  95. {
  96. int ret = 0;
  97. // First copy ints from array
  98. if (remaining > 0)
  99. {
  100. int nr = Math.min(remaining, len);
  101. System.arraycopy(array, arrayIndex, dst, pos, nr);
  102. arrayIndex += nr;
  103. remaining -= nr;
  104. pos += nr;
  105. len -= nr;
  106. ret += nr;
  107. if (len == 0)
  108. {
  109. if (remaining == 0 && !tb.hasRemaining())
  110. copyFileIntoBuffer();
  111. return ret;
  112. }
  113. }
  114. // Now copy ints directly from buffer
  115. while (tb.remaining() < len)
  116. {
  117. int nr = tb.remaining();
  118. tb.get(dst, pos, nr);
  119. pos += nr;
  120. len -= nr;
  121. ret += nr;
  122. if (!copyFileIntoBuffer())
  123. return ret;
  124. }
  125. tb.get(dst, pos, len);
  126. ret += len;
  127. // Discard array
  128. remaining = 0;
  129. return ret;
  130. }
  131. public int get(int index, double[] dst) throws IOException
  132. {
  133. return get(index, dst, 0, dst.length);
  134. }
  135. public final int get(int index, double[] dst, int offset, int len) throws IOException
  136. {
  137. int relIndex = index - startBufferIndex;
  138. if (relIndex < 0 || relIndex >= tb.limit())
  139. {
  140. moveBufferTo(index * ELEMENT_SIZE);
  141. if (remaining < 0)
  142. throw new IndexOutOfBoundsException();
  143. relIndex = index - startBufferIndex;
  144. }
  145. // Change buffer position and discard array
  146. ((Buffer)tb).position(relIndex);
  147. remaining = 0;
  148. return get(dst, offset, len);
  149. }
  150. private boolean copyFileIntoBuffer() throws IOException
  151. {
  152. //Buffer cast is needed with Java 9 (see https://jira.mongodb.org/browse/JAVA-2559)
  153. Buffer tbb = tb;
  154. tbb.clear();
  155. ((Buffer)bb).clear();
  156. int nr;
  157. startBufferIndex = (int) (fc.position() / ELEMENT_SIZE);
  158. do
  159. {
  160. nr = fc.read(bb);
  161. } while (nr == 0);
  162. if (nr == -1)
  163. {
  164. remaining = -1;
  165. return false;
  166. }
  167. nr /= ELEMENT_SIZE;
  168. tbb.position(0);
  169. tbb.limit(nr);
  170. // Discard array
  171. remaining = 0;
  172. return true;
  173. }
  174. private void copyBufferIntoArray() throws IOException
  175. {
  176. if (!tb.hasRemaining())
  177. {
  178. if (!copyFileIntoBuffer())
  179. return;
  180. }
  181. if (array == null)
  182. array = new double[Math.min((int) numberOfElements, ARRAY_SIZE)];
  183. arrayIndex = 0;
  184. remaining = Math.min(tb.remaining(), array.length);
  185. tb.get(array, 0, remaining);
  186. }
  187. public boolean isEOF()
  188. {
  189. return remaining < 0;
  190. }
  191. public void close()
  192. {
  193. try
  194. {
  195. fc.close();
  196. }
  197. catch (IOException ex) { /* Do not care */ }
  198. }
  199. }