/driver-core/src/main/com/mongodb/connection/CompositeByteBuf.java

http://github.com/mongodb/mongo-java-driver · Java · 343 lines · 277 code · 51 blank · 15 comment · 28 complexity · fec63e0c0da0e032c2364c0cb368157a MD5 · raw file

  1. /*
  2. * Copyright 2015 MongoDB, Inc.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package com.mongodb.connection;
  17. import org.bson.ByteBuf;
  18. import java.nio.ByteBuffer;
  19. import java.nio.ByteOrder;
  20. import java.util.ArrayList;
  21. import java.util.List;
  22. import java.util.concurrent.atomic.AtomicInteger;
  23. import static java.lang.String.format;
  24. import static org.bson.assertions.Assertions.isTrueArgument;
  25. import static org.bson.assertions.Assertions.notNull;
  26. class CompositeByteBuf implements ByteBuf {
  27. private final List<Component> components;
  28. private final AtomicInteger referenceCount = new AtomicInteger(1);
  29. private int position;
  30. private int limit;
  31. public CompositeByteBuf(final List<ByteBuf> buffers) {
  32. notNull("buffers", buffers);
  33. isTrueArgument("buffer list not empty", !buffers.isEmpty());
  34. components = new ArrayList<Component>(buffers.size());
  35. int offset = 0;
  36. for (ByteBuf cur : buffers) {
  37. Component component = new Component(cur.duplicate().order(ByteOrder.LITTLE_ENDIAN), offset);
  38. components.add(component);
  39. offset = component.endOffset;
  40. }
  41. limit = components.get(components.size() - 1).endOffset;
  42. }
  43. CompositeByteBuf(final CompositeByteBuf from) {
  44. components = from.components;
  45. position = from.position();
  46. limit = from.limit();
  47. }
  48. @Override
  49. public ByteBuf order(final ByteOrder byteOrder) {
  50. if (byteOrder == ByteOrder.BIG_ENDIAN) {
  51. throw new UnsupportedOperationException(format("Only %s is supported", ByteOrder.BIG_ENDIAN));
  52. }
  53. return this;
  54. }
  55. @Override
  56. public int capacity() {
  57. return components.get(components.size() - 1).endOffset;
  58. }
  59. @Override
  60. public int remaining() {
  61. return limit() - position();
  62. }
  63. @Override
  64. public boolean hasRemaining() {
  65. return remaining() > 0;
  66. }
  67. @Override
  68. public int position() {
  69. return position;
  70. }
  71. @Override
  72. public ByteBuf position(final int newPosition) {
  73. if (newPosition < 0 || newPosition > limit) {
  74. throw new IndexOutOfBoundsException(format("%d is out of bounds", newPosition));
  75. }
  76. position = newPosition;
  77. return this;
  78. }
  79. @Override
  80. public ByteBuf clear() {
  81. position = 0;
  82. limit = capacity();
  83. return this;
  84. }
  85. @Override
  86. public int limit() {
  87. return limit;
  88. }
  89. @Override
  90. public byte get() {
  91. checkIndex(position);
  92. position += 1;
  93. return get(position - 1);
  94. }
  95. @Override
  96. public byte get(final int index) {
  97. checkIndex(index);
  98. Component component = findComponent(index);
  99. return component.buffer.get(index - component.offset);
  100. }
  101. @Override
  102. public ByteBuf get(final byte[] bytes) {
  103. checkIndex(position, bytes.length);
  104. position += bytes.length;
  105. return get(position - bytes.length, bytes);
  106. }
  107. @Override
  108. public ByteBuf get(final int index, final byte[] bytes) {
  109. return get(index, bytes, 0, bytes.length);
  110. }
  111. @Override
  112. public ByteBuf get(final byte[] bytes, final int offset, final int length) {
  113. checkIndex(position, length);
  114. position += length;
  115. return get(position - length, bytes, offset, length);
  116. }
  117. @Override
  118. public ByteBuf get(final int index, final byte[] bytes, final int offset, final int length) {
  119. checkDstIndex(index, length, offset, bytes.length);
  120. int i = findComponentIndex(index);
  121. int curIndex = index;
  122. int curOffset = offset;
  123. int curLength = length;
  124. while (curLength > 0) {
  125. Component c = components.get(i);
  126. int localLength = Math.min(curLength, c.buffer.capacity() - (curIndex - c.offset));
  127. c.buffer.get(curIndex - c.offset, bytes, curOffset, localLength);
  128. curIndex += localLength;
  129. curOffset += localLength;
  130. curLength -= localLength;
  131. i++;
  132. }
  133. return this;
  134. }
  135. @Override
  136. public long getLong() {
  137. position += 8;
  138. return getLong(position - 8);
  139. }
  140. @Override
  141. public long getLong(final int index) {
  142. checkIndex(index, 8);
  143. Component component = findComponent(index);
  144. if (index + 8 <= component.endOffset) {
  145. return component.buffer.getLong(index - component.offset);
  146. } else {
  147. return getInt(index) & 0xFFFFFFFFL | (getInt(index + 4) & 0xFFFFFFFFL) << 32;
  148. }
  149. }
  150. @Override
  151. public double getDouble() {
  152. position += 8;
  153. return getDouble(position - 8);
  154. }
  155. @Override
  156. public double getDouble(final int index) {
  157. return Double.longBitsToDouble(getLong(index));
  158. }
  159. @Override
  160. public int getInt() {
  161. position += 4;
  162. return getInt(position - 4);
  163. }
  164. @Override
  165. public int getInt(final int index) {
  166. checkIndex(index, 4);
  167. Component component = findComponent(index);
  168. if (index + 4 <= component.endOffset) {
  169. return component.buffer.getInt(index - component.offset);
  170. } else {
  171. return getShort(index) & 0xFFFF | (getShort(index + 2) & 0xFFFF) << 16;
  172. }
  173. }
  174. private int getShort(final int index) {
  175. checkIndex(index, 2);
  176. return (short) (get(index) & 0xff | (get(index + 1) & 0xff) << 8);
  177. }
  178. @Override
  179. public byte[] array() {
  180. throw new UnsupportedOperationException("Not implemented yet!");
  181. }
  182. @Override
  183. public ByteBuf limit(final int newLimit) {
  184. if (newLimit < 0 || newLimit > capacity()) {
  185. throw new IndexOutOfBoundsException(format("%d is out of bounds", newLimit));
  186. }
  187. this.limit = newLimit;
  188. return this;
  189. }
  190. @Override
  191. public ByteBuf put(final int index, final byte b) {
  192. throw new UnsupportedOperationException();
  193. }
  194. @Override
  195. public ByteBuf put(final byte[] src, final int offset, final int length) {
  196. throw new UnsupportedOperationException();
  197. }
  198. @Override
  199. public ByteBuf put(final byte b) {
  200. throw new UnsupportedOperationException();
  201. }
  202. @Override
  203. public ByteBuf flip() {
  204. throw new UnsupportedOperationException();
  205. }
  206. @Override
  207. public ByteBuf asReadOnly() {
  208. throw new UnsupportedOperationException();
  209. }
  210. @Override
  211. public ByteBuf duplicate() {
  212. return new CompositeByteBuf(this);
  213. }
  214. @Override
  215. public ByteBuffer asNIO() {
  216. if (components.size() == 1) {
  217. ByteBuffer byteBuffer = components.get(0).buffer.asNIO().duplicate();
  218. byteBuffer.position(position).limit(limit);
  219. return byteBuffer;
  220. } else {
  221. byte[] bytes = new byte[remaining()];
  222. get(position, bytes, 0, bytes.length);
  223. return ByteBuffer.wrap(bytes);
  224. }
  225. }
  226. @Override
  227. public int getReferenceCount() {
  228. return referenceCount.get();
  229. }
  230. @Override
  231. public ByteBuf retain() {
  232. if (referenceCount.incrementAndGet() == 1) {
  233. referenceCount.decrementAndGet();
  234. throw new IllegalStateException("Attempted to increment the reference count when it is already 0");
  235. }
  236. return this;
  237. }
  238. @Override
  239. public void release() {
  240. if (referenceCount.decrementAndGet() < 0) {
  241. referenceCount.incrementAndGet();
  242. throw new IllegalStateException("Attempted to decrement the reference count below 0");
  243. }
  244. }
  245. private Component findComponent(final int index) {
  246. return components.get(findComponentIndex(index));
  247. }
  248. private int findComponentIndex(final int index) {
  249. for (int i = components.size() - 1; i >= 0; i--) {
  250. Component cur = components.get(i);
  251. if (index >= cur.offset) {
  252. return i;
  253. }
  254. }
  255. throw new IndexOutOfBoundsException(format("%d is out of bounds", index));
  256. }
  257. private void checkIndex(final int index) {
  258. ensureAccessible();
  259. if (index < 0 || index >= capacity()) {
  260. throw new IndexOutOfBoundsException(format("index: %d (expected: range(0, %d))", index, capacity()));
  261. }
  262. }
  263. private void checkIndex(final int index, final int fieldLength) {
  264. ensureAccessible();
  265. if (index < 0 || index > capacity() - fieldLength) {
  266. throw new IndexOutOfBoundsException(format("index: %d, length: %d (expected: range(0, %d))", index, fieldLength, capacity()));
  267. }
  268. }
  269. private void checkDstIndex(final int index, final int length, final int dstIndex, final int dstCapacity) {
  270. checkIndex(index, length);
  271. if (dstIndex < 0 || dstIndex > dstCapacity - length) {
  272. throw new IndexOutOfBoundsException(format("dstIndex: %d, length: %d (expected: range(0, %d))", dstIndex, length, dstCapacity));
  273. }
  274. }
  275. private void ensureAccessible() {
  276. if (referenceCount.get() == 0) {
  277. throw new IllegalStateException("Reference count is 0");
  278. }
  279. }
  280. private static final class Component {
  281. private final ByteBuf buffer;
  282. private final int length;
  283. private final int offset;
  284. private final int endOffset;
  285. Component(final ByteBuf buffer, final int offset) {
  286. this.buffer = buffer;
  287. length = buffer.limit() - buffer.position();
  288. this.offset = offset;
  289. this.endOffset = offset + length;
  290. }
  291. }
  292. }