/src/main/java/org/lwes/serializer/Serializer.java

http://github.com/lwes/lwes-java · Java · 916 lines · 666 code · 109 blank · 141 comment · 62 complexity · f8e8e6cfc108abf975446746bcbe8cdb MD5 · raw file

  1. /*======================================================================*
  2. * Copyright (c) 2008, Yahoo! Inc. All rights reserved. *
  3. * *
  4. * Licensed under the New BSD License (the "License"); you may not use *
  5. * this file except in compliance with the License. Unless required *
  6. * by applicable law or agreed to in writing, software distributed *
  7. * under the License is distributed on an "AS IS" BASIS, WITHOUT *
  8. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. *
  9. * See the License for the specific language governing permissions and *
  10. * limitations under the License. See accompanying LICENSE file. *
  11. *======================================================================*/
  12. package org.lwes.serializer;
  13. import java.math.BigInteger;
  14. import java.net.Inet4Address;
  15. import java.net.InetAddress;
  16. import java.util.ArrayList;
  17. import java.util.BitSet;
  18. import java.util.List;
  19. import org.lwes.Event;
  20. import org.lwes.FieldType;
  21. import org.lwes.util.EncodedString;
  22. import org.lwes.util.IPAddress;
  23. import org.lwes.util.NumberCodec;
  24. /**
  25. * This contains low level type serialization used by the
  26. * rest of the system.
  27. *
  28. * @author Anthony Molinaro
  29. * @author Frank Maritato
  30. */
  31. public class Serializer {
  32. public static int serializeIPV4(Inet4Address value, byte[] bytes, int offset) {
  33. byte[] bs = value.getAddress();
  34. int i = offset;
  35. for (byte b : bs) {
  36. bytes[i++] = b;
  37. }
  38. return offset - i;
  39. }
  40. public static int serializeDOUBLE(Double value, byte[] bytes, int offset) {
  41. long j = Double.doubleToLongBits(value);
  42. bytes[offset + 7] = (byte) (j >>> 0);
  43. bytes[offset + 6] = (byte) (j >>> 8);
  44. bytes[offset + 5] = (byte) (j >>> 16);
  45. bytes[offset + 4] = (byte) (j >>> 24);
  46. bytes[offset + 3] = (byte) (j >>> 32);
  47. bytes[offset + 2] = (byte) (j >>> 40);
  48. bytes[offset + 1] = (byte) (j >>> 48);
  49. bytes[offset + 0] = (byte) (j >>> 56);
  50. return 8;
  51. }
  52. public static int serializeFLOAT(Float value, byte[] bytes, int offset) {
  53. int i = Float.floatToIntBits(value);
  54. bytes[offset + 3] = (byte) (i >>> 0);
  55. bytes[offset + 2] = (byte) (i >>> 8);
  56. bytes[offset + 1] = (byte) (i >>> 16);
  57. bytes[offset + 0] = (byte) (i >>> 24);
  58. return 4;
  59. }
  60. public static int serializeBYTE(byte value, byte[] bytes, int offset) {
  61. bytes[offset] = value;
  62. return 1;
  63. }
  64. public static int serializeUBYTE(short value, byte[] bytes, int offset) throws IllegalArgumentException {
  65. if (value < 0 || value > 255) {
  66. throw new IllegalArgumentException("Unsigned byte " + value + " out of range 0..255");
  67. }
  68. bytes[offset] = (byte) (value & 0xff);
  69. return 1;
  70. }
  71. public static int serializeBOOLEAN(boolean value, byte[] bytes, int offset) {
  72. bytes[offset] = (byte) (value ? 1 : 0);
  73. return 1;
  74. }
  75. public static int serializeUINT16(int value, byte[] bytes, int offset) {
  76. return serializeINT16((short) value, bytes, offset);
  77. }
  78. public static int serializeINT16(short value, byte[] bytes, int offset) {
  79. bytes[offset + 1] = (byte) (value >>> 0);
  80. bytes[offset + 0] = (byte) (value >>> 8);
  81. return 2;
  82. }
  83. public static int serializeUINT32(long anUnsignedInt, byte[] bytes, int offset) {
  84. bytes[offset] = (byte) ((anUnsignedInt & 0xff000000) >> 24);
  85. bytes[offset + 1] = (byte) ((anUnsignedInt & 0x00ff0000) >> 16);
  86. bytes[offset + 2] = (byte) ((anUnsignedInt & 0x0000ff00) >> 8);
  87. bytes[offset + 3] = (byte) ((anUnsignedInt & 0x000000ff) >> 0);
  88. return (4);
  89. }
  90. public static int serializeINT32(int value, byte[] bytes, int offset) {
  91. bytes[offset + 3] = (byte) (value >>> 0);
  92. bytes[offset + 2] = (byte) (value >>> 8);
  93. bytes[offset + 1] = (byte) (value >>> 16);
  94. bytes[offset + 0] = (byte) (value >>> 24);
  95. return 4;
  96. }
  97. public static int serializeINT64(long value, byte[] bytes, int offset) {
  98. bytes[offset + 7] = (byte) (value >>> 0);
  99. bytes[offset + 6] = (byte) (value >>> 8);
  100. bytes[offset + 5] = (byte) (value >>> 16);
  101. bytes[offset + 4] = (byte) (value >>> 24);
  102. bytes[offset + 3] = (byte) (value >>> 32);
  103. bytes[offset + 2] = (byte) (value >>> 40);
  104. bytes[offset + 1] = (byte) (value >>> 48);
  105. bytes[offset + 0] = (byte) (value >>> 56);
  106. return 8;
  107. }
  108. public static int serializeUINT64(long anInt, byte[] bytes, int offset) {
  109. NumberCodec.encodeLongUnchecked(anInt, bytes, offset);
  110. return (8);
  111. }
  112. public static int serializeUINT64(BigInteger anInt, byte[] bytes, int offset) {
  113. // TODO: write a BigInteger serialization method
  114. NumberCodec.encodeLongUnchecked(anInt.longValue(), bytes, offset);
  115. return (8);
  116. }
  117. /**
  118. * @deprecated
  119. */
  120. @Deprecated
  121. public static int serializeSTRING(String aString, byte[] bytes, int offset) {
  122. return serializeSTRING(aString, bytes, offset, Event.DEFAULT_ENCODING);
  123. }
  124. public static int serializeSTRING(String aString, byte[] bytes, int offset,
  125. short encoding) {
  126. byte[] stringBytes =
  127. EncodedString.getBytes(aString, Event.ENCODING_STRINGS[encoding]);
  128. int length = stringBytes.length;
  129. if (length < 65535 && length >= 0) {
  130. offset += serializeUINT16(length, bytes, offset);
  131. System.arraycopy(stringBytes, 0, bytes, offset, length);
  132. return (length + 2);
  133. }
  134. return 0;
  135. }
  136. /**
  137. * String arrays are serialized as follows:
  138. * &lt;array_name_len&gt;&lt;array_name_bytes&gt;&lt;type&gt;
  139. * &lt;array_length&gt;&lt;serialized_type_1&gt;...&lt;serialized_type_n&gt;
  140. *
  141. * @param value
  142. * @param bytes
  143. * @param offset
  144. * @param encoding
  145. * @return the offset
  146. */
  147. public static int serializeStringArray(String[] value,
  148. byte[] bytes,
  149. int offset,
  150. short encoding) {
  151. int numbytes = 0;
  152. int offsetStart = offset;
  153. numbytes = serializeUINT16(value.length, bytes, offset);
  154. offset += numbytes;
  155. for (String s : value) {
  156. numbytes = serializeSTRING(s, bytes, offset, encoding);
  157. offset += numbytes;
  158. }
  159. return (offset - offsetStart);
  160. }
  161. public static int serializeIPADDRArray(IPAddress[] value,
  162. byte[] bytes,
  163. int offset) {
  164. int numbytes = 0;
  165. int offsetStart = offset;
  166. numbytes = serializeUINT16(value.length, bytes, offset);
  167. offset += numbytes;
  168. for (IPAddress s : value) {
  169. numbytes = serializeIPADDR(s, bytes, offset);
  170. offset += numbytes;
  171. }
  172. return (offset - offsetStart);
  173. }
  174. public static int serializeInt16Array(short[] value,
  175. byte[] bytes,
  176. int offset) {
  177. int numbytes = 0;
  178. int offsetStart = offset;
  179. numbytes = serializeUINT16(value.length, bytes, offset);
  180. offset += numbytes;
  181. for (short s : value) {
  182. numbytes = serializeINT16(s, bytes, offset);
  183. offset += numbytes;
  184. }
  185. return (offset - offsetStart);
  186. }
  187. public static int serializeInt32Array(int[] value,
  188. byte[] bytes,
  189. int offset) {
  190. int numbytes = 0;
  191. int offsetStart = offset;
  192. numbytes = serializeUINT16(value.length, bytes, offset);
  193. offset += numbytes;
  194. for (int s : value) {
  195. numbytes = serializeINT32(s, bytes, offset);
  196. offset += numbytes;
  197. }
  198. return (offset - offsetStart);
  199. }
  200. public static int serializeInt64Array(long[] value,
  201. byte[] bytes,
  202. int offset) {
  203. int numbytes = 0;
  204. int offsetStart = offset;
  205. numbytes = serializeUINT16(value.length, bytes, offset);
  206. offset += numbytes;
  207. for (long s : value) {
  208. numbytes = serializeINT64(s, bytes, offset);
  209. offset += numbytes;
  210. }
  211. return (offset - offsetStart);
  212. }
  213. public static int serializeUInt16Array(int[] value,
  214. byte[] bytes,
  215. int offset) {
  216. int numbytes = 0;
  217. int offsetStart = offset;
  218. numbytes = serializeUINT16(value.length, bytes, offset);
  219. offset += numbytes;
  220. for (int s : value) {
  221. numbytes = serializeUINT16(s, bytes, offset);
  222. offset += numbytes;
  223. }
  224. return (offset - offsetStart);
  225. }
  226. public static int serializeUInt32Array(long[] value,
  227. byte[] bytes,
  228. int offset) {
  229. int numbytes = 0;
  230. int offsetStart = offset;
  231. numbytes = serializeUINT16(value.length, bytes, offset);
  232. offset += numbytes;
  233. for (long s : value) {
  234. numbytes = serializeUINT32(s, bytes, offset);
  235. offset += numbytes;
  236. }
  237. return (offset - offsetStart);
  238. }
  239. public static int serializeUInt64Array(BigInteger[] value,
  240. byte[] bytes,
  241. int offset) {
  242. int numbytes = 0;
  243. int offsetStart = offset;
  244. numbytes = serializeUINT16(value.length, bytes, offset);
  245. offset += numbytes;
  246. for (BigInteger s : value) {
  247. numbytes = serializeUINT64(s, bytes, offset);
  248. offset += numbytes;
  249. }
  250. return (offset - offsetStart);
  251. }
  252. public static int serializeBooleanArray(boolean[] value,
  253. byte[] bytes,
  254. int offset) {
  255. int numbytes = 0;
  256. int offsetStart = offset;
  257. numbytes = serializeUINT16(value.length, bytes, offset);
  258. offset += numbytes;
  259. for (boolean s : value) {
  260. numbytes = serializeBOOLEAN(s, bytes, offset);
  261. offset += numbytes;
  262. }
  263. return (offset - offsetStart);
  264. }
  265. public static int serializeByteArray(byte[] value,
  266. byte[] bytes,
  267. int offset) {
  268. int numbytes = 0;
  269. int offsetStart = offset;
  270. numbytes = serializeUINT16(value.length, bytes, offset);
  271. offset += numbytes;
  272. for (byte s : value) {
  273. numbytes = serializeBYTE(s, bytes, offset);
  274. offset += numbytes;
  275. }
  276. return (offset - offsetStart);
  277. }
  278. public static int serializeDoubleArray(double[] value, byte[] bytes, int offset) {
  279. int numbytes = 0;
  280. int offsetStart = offset;
  281. numbytes = serializeUINT16(value.length, bytes, offset);
  282. offset += numbytes;
  283. for (double s : value) {
  284. numbytes = serializeDOUBLE(s, bytes, offset);
  285. offset += numbytes;
  286. }
  287. return (offset - offsetStart);
  288. }
  289. public static int serializeFloatArray(float[] value, byte[] bytes, int offset) {
  290. int numbytes = 0;
  291. int offsetStart = offset;
  292. numbytes = serializeUINT16(value.length, bytes, offset);
  293. offset += numbytes;
  294. for (float s : value) {
  295. numbytes = serializeFLOAT(s, bytes, offset);
  296. offset += numbytes;
  297. }
  298. return (offset - offsetStart);
  299. }
  300. public static int serializeEVENTWORD(String aString, byte[] bytes, int offset) {
  301. return serializeEVENTWORD(aString, bytes, offset, Event.DEFAULT_ENCODING);
  302. }
  303. private static int serializeEVENTWORD(String aString,
  304. byte[] bytes,
  305. int offset,
  306. short encoding) {
  307. byte[] stringBytes =
  308. EncodedString.getBytes(aString, Event.ENCODING_STRINGS[encoding]);
  309. int length = stringBytes.length;
  310. if (0 <= length && length <= 255) {
  311. offset += serializeUBYTE((short) length, bytes, offset);
  312. System.arraycopy(stringBytes, 0, bytes, offset, length);
  313. return (length + 1);
  314. }
  315. return 0;
  316. }
  317. public static int serializeATTRIBUTEWORD(String aString, byte[] bytes,
  318. int offset) {
  319. return serializeEVENTWORD(aString, bytes, offset, Event.DEFAULT_ENCODING);
  320. }
  321. /**
  322. * Serialize IPAddress in *reverse* network order. Don't use this.
  323. *
  324. * @param anIPAddress the ip address to serialize
  325. * @param bytes the byte array to modify
  326. * @param offset what index in the array to start at
  327. * @return how many bytes were set in the array
  328. * @deprecated
  329. */
  330. @Deprecated
  331. public static int serializeIPADDR(IPAddress anIPAddress, byte[] bytes, int offset) {
  332. byte[] inetaddr = anIPAddress.getInetAddressAsBytes();
  333. bytes[offset + 3] = inetaddr[0];
  334. bytes[offset + 2] = inetaddr[1];
  335. bytes[offset + 1] = inetaddr[2];
  336. bytes[offset] = inetaddr[3];
  337. return (4);
  338. }
  339. /**
  340. * Serializes an ip address in *reverse* network order. Don't use this.
  341. *
  342. * @param anIPAddress the ip address to serialize
  343. * @param bytes the byte array to modify
  344. * @param offset what index in the array to start at
  345. * @return how many bytes were set in the array
  346. * @deprecated
  347. */
  348. @Deprecated
  349. public static int serializeIPADDR(InetAddress anIPAddress, byte[] bytes, int offset) {
  350. byte[] inetaddr = anIPAddress.getAddress();
  351. bytes[offset + 3] = inetaddr[0];
  352. bytes[offset + 2] = inetaddr[1];
  353. bytes[offset + 1] = inetaddr[2];
  354. bytes[offset] = inetaddr[3];
  355. return (4);
  356. }
  357. /**
  358. * Serialize InetAddress in network order.
  359. *
  360. * @param ip the ip address to serialize
  361. * @param bytes the byte array to modify
  362. * @param offset what index in the array to start at
  363. * @return how many bytes were set in the array
  364. */
  365. public static int serializeIPV4(InetAddress ip, byte[] bytes, int offset) {
  366. byte[] inetaddr = ip.getAddress();
  367. bytes[offset] = inetaddr[0];
  368. bytes[offset + 1] = inetaddr[1];
  369. bytes[offset + 2] = inetaddr[2];
  370. bytes[offset + 3] = inetaddr[3];
  371. return (4);
  372. }
  373. public static int serializeValue(FieldType type,
  374. Object data,
  375. short encoding,
  376. byte[] bytes,
  377. final int offset) {
  378. switch (type) {
  379. case BYTE:
  380. return Serializer.serializeBYTE((Byte) data, bytes, offset);
  381. case BOOLEAN:
  382. return Serializer.serializeBOOLEAN((Boolean) data, bytes, offset);
  383. case UINT16:
  384. return Serializer.serializeUINT16((Integer) data, bytes, offset);
  385. case INT16:
  386. return Serializer.serializeINT16((Short) data, bytes, offset);
  387. case UINT32:
  388. return Serializer.serializeUINT32((Long) data, bytes, offset);
  389. case INT32:
  390. return Serializer.serializeINT32((Integer) data, bytes, offset);
  391. case UINT64:
  392. return Serializer.serializeUINT64((BigInteger) data, bytes, offset);
  393. case INT64:
  394. return Serializer.serializeINT64((Long) data, bytes, offset);
  395. case STRING:
  396. return Serializer.serializeSTRING(((String) data), bytes, offset, encoding);
  397. case DOUBLE:
  398. return Serializer.serializeDOUBLE(((Double) data), bytes, offset);
  399. case FLOAT:
  400. return Serializer.serializeFLOAT(((Float) data), bytes, offset);
  401. case IPADDR:
  402. return Serializer.serializeIPADDR(((IPAddress) data), bytes, offset);
  403. case STRING_ARRAY:
  404. return Serializer.serializeStringArray
  405. (((String[]) data), bytes, offset, encoding);
  406. case NUINT32_ARRAY:
  407. return Serializer.serializeNUInt32Array((Long[]) data, bytes, offset);
  408. case NINT32_ARRAY:
  409. return Serializer.serializeNInt32Array((Integer[]) data, bytes, offset);
  410. case NUINT16_ARRAY:
  411. return Serializer.serializeNUInt16Array((Integer[]) data, bytes, offset);
  412. case NINT16_ARRAY:
  413. return Serializer.serializeNInt16Array((Short[]) data, bytes, offset);
  414. case NINT64_ARRAY:
  415. return Serializer.serializeNInt64Array((Long[]) data, bytes, offset);
  416. case NDOUBLE_ARRAY:
  417. return Serializer.serializeNDoubleArray((Double[]) data, bytes, offset);
  418. case NFLOAT_ARRAY:
  419. return Serializer.serializeNFloatArray((Float[]) data, bytes, offset);
  420. case INT16_ARRAY:
  421. return Serializer.serializeInt16Array((short[]) data, bytes, offset);
  422. case INT32_ARRAY:
  423. return Serializer.serializeInt32Array((int[]) data, bytes, offset);
  424. case INT64_ARRAY:
  425. return Serializer.serializeInt64Array((long[]) data, bytes, offset);
  426. case UINT16_ARRAY:
  427. return Serializer.serializeUInt16Array((int[]) data, bytes, offset);
  428. case UINT32_ARRAY:
  429. return Serializer.serializeUInt32Array((long[]) data, bytes, offset);
  430. case UINT64_ARRAY:
  431. return Serializer.serializeUInt64Array((BigInteger[]) data, bytes, offset);
  432. case NUINT64_ARRAY:
  433. return Serializer.serializeNUInt64Array((BigInteger[]) data, bytes, offset);
  434. case NBOOLEAN_ARRAY:
  435. return Serializer.serializeNBooleanArray((Boolean[]) data, bytes, offset);
  436. case BOOLEAN_ARRAY:
  437. return Serializer.serializeBooleanArray((boolean[]) data, bytes, offset);
  438. case NBYTE_ARRAY:
  439. return Serializer.serializeNByteArray((Byte[]) data, bytes, offset);
  440. case BYTE_ARRAY:
  441. return Serializer.serializeByteArray((byte[]) data, bytes, offset);
  442. case DOUBLE_ARRAY:
  443. return Serializer.serializeDoubleArray((double[]) data, bytes, offset);
  444. case FLOAT_ARRAY:
  445. return Serializer.serializeFloatArray((float[]) data, bytes, offset);
  446. case IP_ADDR_ARRAY:
  447. return Serializer.serializeIPADDRArray((IPAddress[]) data, bytes, offset);
  448. case NSTRING_ARRAY:
  449. return Serializer.serializeNStringArray((String[]) data, bytes, offset, encoding);
  450. }
  451. throw new IllegalArgumentException("Unknown BaseType token: " + type);
  452. }
  453. /**
  454. * Pack as many bitset elements into one byte as possible.
  455. *
  456. * @param bitSet BitSet that indicates which indexes in an array are not null.
  457. * @param data the byte array to write the packed version of the bitSet to
  458. * @param offset the index to start writing to data
  459. * @return the number of bytes written to data
  460. */
  461. public static int serializeBitSet(BitSet bitSet, int arrayLength, byte[] data, int offset) {
  462. int offsetStart = offset;
  463. offset += serializeUINT16((short) arrayLength, data, offset);
  464. int lastIndex = 0;
  465. final int paddedBitSetLength = 8 * (int) Math.ceil(arrayLength/ 8.0);
  466. for (int i = 0, c=0; i < paddedBitSetLength; i++,c++) {
  467. int index = i >>> 3;
  468. if (index != lastIndex) {
  469. c = 0;
  470. lastIndex = index;
  471. }
  472. if (bitSet.get(i)) {
  473. data[offset + index] |= (1 << c);
  474. } else {
  475. data[offset + index] &= ~(1 << c);
  476. }
  477. }
  478. // Set the offset to how many bits we had to use
  479. offset += (int) Math.ceil(arrayLength / 8.0);
  480. assert Deserializer.deserializeBitSet(new DeserializerState(offsetStart), data).equals(bitSet);
  481. return (offset - offsetStart);
  482. }
  483. /**
  484. * For serializing arrays that can contain nulls. For strings, we need to figure out how long each string is
  485. * in order to create the temporary array to write into
  486. *
  487. * @param data array to serialize
  488. * @param bytes byte array to write to
  489. * @param offset index in byte array to start at
  490. * @param encoding encoding to use for strings
  491. * @return number of bytes written
  492. */
  493. public static int serializeNStringArray(String[] data, byte[] bytes, int offset, short encoding) {
  494. int numbytes = 0;
  495. int offsetStart = offset;
  496. // Number of items in the array
  497. numbytes = serializeUINT16((short) data.length, bytes, offset);
  498. offset += numbytes;
  499. List<byte[]> tmp = new ArrayList<byte[]>(data.length);
  500. // use a bitset to determine which indexes have values and which are null.
  501. // serialize the actual values into a temporary byte array
  502. BitSet bitSet = new BitSet(data.length);
  503. int i = 0;
  504. for (String s : data) {
  505. if (s != null) {
  506. byte[] stringBytes = EncodedString.getBytes(s, Event.ENCODING_STRINGS[encoding]);
  507. byte[] tmpArr = new byte[stringBytes.length + 2];
  508. serializeUINT16(stringBytes.length, tmpArr, 0);
  509. System.arraycopy(stringBytes, 0, tmpArr, 2, stringBytes.length);
  510. tmp.add(i, tmpArr);
  511. bitSet.set(i);
  512. }
  513. else {
  514. tmp.add(i, null);
  515. }
  516. i++;
  517. }
  518. // Write the bitset first to ease with deserialization
  519. offset += serializeBitSet(bitSet, data.length, bytes, offset);
  520. // Now write the values
  521. for (byte[] a : tmp) {
  522. if (a != null) {
  523. System.arraycopy(a, 0, bytes, offset, a.length);
  524. offset += a.length;
  525. }
  526. }
  527. return (offset - offsetStart);
  528. }
  529. /**
  530. * For serializing arrays that can contain nulls, blahblah
  531. *
  532. * @param data array to serialize
  533. * @param bytes byte array to write to
  534. * @param offset index in byte array to start at
  535. * @return number of bytes written
  536. */
  537. public static int serializeNFloatArray(Float[] data, byte[] bytes, int offset) {
  538. int numbytes = 0;
  539. int offsetStart = offset;
  540. // Number of items in the array
  541. numbytes = serializeUINT16((short) data.length, bytes, offset);
  542. offset += numbytes;
  543. byte[] tmp = new byte[4 * data.length];
  544. int tmpOffset = 0;
  545. // use a bitset to determine which indexes have values and which are null.
  546. // serialize the actual values into a temporary byte array
  547. BitSet bitSet = new BitSet(data.length);
  548. int i = 0;
  549. for (Float s : data) {
  550. if (s != null) {
  551. numbytes = serializeFLOAT(s, tmp, tmpOffset);
  552. tmpOffset += numbytes;
  553. bitSet.set(i);
  554. }
  555. i++;
  556. }
  557. // Write the bitset first to ease with deserialization
  558. offset += serializeBitSet(bitSet, data.length, bytes, offset);
  559. // Now write the float values
  560. System.arraycopy(tmp, 0, bytes, offset, tmpOffset);
  561. offset += tmpOffset;
  562. return (offset - offsetStart);
  563. }
  564. public static int serializeNByteArray(Byte[] data, byte[] bytes, int offset) {
  565. int numbytes = 0;
  566. int offsetStart = offset;
  567. // Number of items in the array
  568. numbytes = serializeUINT16((short) data.length, bytes, offset);
  569. offset += numbytes;
  570. byte[] tmp = new byte[data.length];
  571. int tmpOffset = 0;
  572. // use a bitset to determine which indexes have values and which are null.
  573. // serialize the actual values into a temporary byte array
  574. BitSet bitSet = new BitSet(data.length);
  575. int i = 0;
  576. for (Byte s : data) {
  577. if (s != null) {
  578. numbytes = serializeBYTE(s, tmp, tmpOffset);
  579. tmpOffset += numbytes;
  580. bitSet.set(i);
  581. }
  582. i++;
  583. }
  584. // Write the bitset first to ease with deserialization
  585. offset += serializeBitSet(bitSet, data.length, bytes, offset);
  586. // Now write the float values
  587. System.arraycopy(tmp, 0, bytes, offset, tmpOffset);
  588. offset += tmpOffset;
  589. return (offset - offsetStart);
  590. }
  591. public static int serializeNBooleanArray(Boolean[] data, byte[] bytes, int offset) {
  592. int numbytes = 0;
  593. int offsetStart = offset;
  594. // Number of items in the array
  595. numbytes = serializeUINT16((short) data.length, bytes, offset);
  596. offset += numbytes;
  597. byte[] tmp = new byte[data.length];
  598. int tmpOffset = 0;
  599. // use a bitset to determine which indexes have values and which are null.
  600. // serialize the actual values into a temporary byte array
  601. BitSet bitSet = new BitSet(data.length);
  602. int i = 0;
  603. for (Boolean s : data) {
  604. if (s != null) {
  605. numbytes = serializeBOOLEAN(s, tmp, tmpOffset);
  606. tmpOffset += numbytes;
  607. bitSet.set(i);
  608. }
  609. i++;
  610. }
  611. // Write the bitset first to ease with deserialization
  612. offset += serializeBitSet(bitSet, data.length, bytes, offset);
  613. // Now write the float values
  614. System.arraycopy(tmp, 0, bytes, offset, tmpOffset);
  615. offset += tmpOffset;
  616. return (offset - offsetStart);
  617. }
  618. public static int serializeNDoubleArray(Double[] data, byte[] bytes, int offset) {
  619. int numbytes = 0;
  620. int offsetStart = offset;
  621. // Number of items in the array
  622. numbytes = serializeUINT16((short) data.length, bytes, offset);
  623. offset += numbytes;
  624. byte[] tmp = new byte[8 * data.length];
  625. int tmpOffset = 0;
  626. // use a bitset to determine which indexes have values and which are null.
  627. // serialize the actual values into a temporary byte array
  628. BitSet bitSet = new BitSet(data.length);
  629. int i = 0;
  630. for (Double s : data) {
  631. if (s != null) {
  632. numbytes = serializeDOUBLE(s, tmp, tmpOffset);
  633. tmpOffset += numbytes;
  634. bitSet.set(i);
  635. }
  636. i++;
  637. }
  638. // Write the bitset first to ease with deserialization
  639. offset += serializeBitSet(bitSet, data.length, bytes, offset);
  640. // Now write the float values
  641. System.arraycopy(tmp, 0, bytes, offset, tmpOffset);
  642. offset += tmpOffset;
  643. return (offset - offsetStart);
  644. }
  645. public static int serializeNInt16Array(Short[] data, byte[] bytes, int offset) {
  646. int numbytes = 0;
  647. int offsetStart = offset;
  648. // Number of items in the array
  649. numbytes = serializeUINT16((short) data.length, bytes, offset);
  650. offset += numbytes;
  651. byte[] tmp = new byte[2 * data.length];
  652. int tmpOffset = 0;
  653. // use a bitset to determine which indexes have values and which are null.
  654. // serialize the actual values into a temporary byte array
  655. BitSet bitSet = new BitSet(data.length);
  656. int i = 0;
  657. for (Short s : data) {
  658. if (s != null) {
  659. numbytes = serializeINT16(s, tmp, tmpOffset);
  660. tmpOffset += numbytes;
  661. bitSet.set(i);
  662. }
  663. i++;
  664. }
  665. // Write the bitset first to ease with deserialization
  666. offset += serializeBitSet(bitSet, data.length, bytes, offset);
  667. // Now write the float values
  668. System.arraycopy(tmp, 0, bytes, offset, tmpOffset);
  669. offset += tmpOffset;
  670. return (offset - offsetStart);
  671. }
  672. public static int serializeNUInt16Array(Integer[] data, byte[] bytes, int offset) {
  673. int numbytes = 0;
  674. int offsetStart = offset;
  675. // Number of items in the array
  676. numbytes = serializeUINT16((short) data.length, bytes, offset);
  677. offset += numbytes;
  678. byte[] tmp = new byte[2 * data.length];
  679. int tmpOffset = 0;
  680. // use a bitset to determine which indexes have values and which are null.
  681. // serialize the actual values into a temporary byte array
  682. BitSet bitSet = new BitSet(data.length);
  683. int i = 0;
  684. for (Integer s : data) {
  685. if (s != null) {
  686. numbytes = serializeUINT16(s, tmp, tmpOffset);
  687. tmpOffset += numbytes;
  688. bitSet.set(i);
  689. }
  690. i++;
  691. }
  692. // Write the bitset first to ease with deserialization
  693. offset += serializeBitSet(bitSet, data.length, bytes, offset);
  694. // Now write the float values
  695. System.arraycopy(tmp, 0, bytes, offset, tmpOffset);
  696. offset += tmpOffset;
  697. return (offset - offsetStart);
  698. }
  699. public static int serializeNInt32Array(Integer[] data, byte[] bytes, int offset) {
  700. int numbytes = 0;
  701. int offsetStart = offset;
  702. // Number of items in the array
  703. numbytes = serializeUINT16((short) data.length, bytes, offset);
  704. offset += numbytes;
  705. byte[] tmp = new byte[4 * data.length];
  706. int tmpOffset = 0;
  707. // use a bitset to determine which indexes have values and which are null.
  708. // serialize the actual values into a temporary byte array
  709. BitSet bitSet = new BitSet(data.length);
  710. int i = 0;
  711. for (Integer s : data) {
  712. if (s != null) {
  713. numbytes = serializeINT32(s, tmp, tmpOffset);
  714. tmpOffset += numbytes;
  715. bitSet.set(i);
  716. }
  717. i++;
  718. }
  719. // Write the bitset first to ease with deserialization
  720. offset += serializeBitSet(bitSet, data.length, bytes, offset);
  721. // Now write the float values
  722. System.arraycopy(tmp, 0, bytes, offset, tmpOffset);
  723. offset += tmpOffset;
  724. return (offset - offsetStart);
  725. }
  726. public static int serializeNUInt32Array(Long[] data, byte[] bytes, int offset) {
  727. int numbytes = 0;
  728. int offsetStart = offset;
  729. // Number of items in the array
  730. numbytes = serializeUINT16((short) data.length, bytes, offset);
  731. offset += numbytes;
  732. byte[] tmp = new byte[4 * data.length];
  733. int tmpOffset = 0;
  734. // use a bitset to determine which indexes have values and which are null.
  735. // serialize the actual values into a temporary byte array
  736. BitSet bitSet = new BitSet(data.length);
  737. int i = 0;
  738. for (Long s : data) {
  739. if (s != null) {
  740. numbytes = serializeUINT32(s, tmp, tmpOffset);
  741. tmpOffset += numbytes;
  742. bitSet.set(i);
  743. }
  744. i++;
  745. }
  746. // Write the bitset first to ease with deserialization
  747. offset += serializeBitSet(bitSet, data.length, bytes, offset);
  748. // Now write the float values
  749. System.arraycopy(tmp, 0, bytes, offset, tmpOffset);
  750. offset += tmpOffset;
  751. return (offset - offsetStart);
  752. }
  753. public static int serializeNInt64Array(Long[] data, byte[] bytes, int offset) {
  754. int numbytes = 0;
  755. int offsetStart = offset;
  756. // Number of items in the array
  757. numbytes = serializeUINT16((short) data.length, bytes, offset);
  758. offset += numbytes;
  759. byte[] tmp = new byte[8 * data.length];
  760. int tmpOffset = 0;
  761. // use a bitset to determine which indexes have values and which are null.
  762. // serialize the actual values into a temporary byte array
  763. BitSet bitSet = new BitSet(data.length);
  764. int i = 0;
  765. for (Long s : data) {
  766. if (s != null) {
  767. numbytes = serializeINT64(s, tmp, tmpOffset);
  768. tmpOffset += numbytes;
  769. bitSet.set(i);
  770. }
  771. i++;
  772. }
  773. // Write the bitset first to ease with deserialization
  774. offset += serializeBitSet(bitSet, data.length, bytes, offset);
  775. // Now write the float values
  776. System.arraycopy(tmp, 0, bytes, offset, tmpOffset);
  777. offset += tmpOffset;
  778. return (offset - offsetStart);
  779. }
  780. public static int serializeNUInt64Array(BigInteger[] data, byte[] bytes, int offset) {
  781. int numbytes = 0;
  782. int offsetStart = offset;
  783. // Number of items in the array
  784. numbytes = serializeUINT16((short) data.length, bytes, offset);
  785. offset += numbytes;
  786. byte[] tmp = new byte[8 * data.length];
  787. int tmpOffset = 0;
  788. // use a bitset to determine which indexes have values and which are null.
  789. // serialize the actual values into a temporary byte array
  790. BitSet bitSet = new BitSet(data.length);
  791. int i = 0;
  792. for (BigInteger s : data) {
  793. if (s != null) {
  794. numbytes = serializeUINT64(s, tmp, tmpOffset);
  795. tmpOffset += numbytes;
  796. bitSet.set(i);
  797. }
  798. i++;
  799. }
  800. // Write the bitset first to ease with deserialization
  801. offset += serializeBitSet(bitSet, data.length, bytes, offset);
  802. // Now write the float values
  803. System.arraycopy(tmp, 0, bytes, offset, tmpOffset);
  804. offset += tmpOffset;
  805. return (offset - offsetStart);
  806. }
  807. }