1 /* 2 * Copyright (C) 2019 The Android Open Source Project 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 17 package com.android.test.protoinputstream; 18 19 import android.util.proto.ProtoInputStream; 20 import android.util.proto.ProtoStream; 21 import android.util.proto.WireTypeMismatchException; 22 23 import com.android.test.protoinputstream.nano.Test; 24 25 import com.google.protobuf.nano.MessageNano; 26 27 import junit.framework.TestCase; 28 29 import java.io.ByteArrayInputStream; 30 import java.io.IOException; 31 import java.io.InputStream; 32 33 public class ProtoInputStreamDoubleTest extends TestCase { 34 35 testRead()36 public void testRead() throws IOException { 37 testRead(0); 38 testRead(1); 39 testRead(5); 40 } 41 testRead(int chunkSize)42 private void testRead(int chunkSize) throws IOException { 43 final long fieldFlags = ProtoStream.FIELD_COUNT_SINGLE | ProtoStream.FIELD_TYPE_DOUBLE; 44 45 final long fieldId1 = fieldFlags | ((long) 1 & 0x0ffffffffL); 46 final long fieldId2 = fieldFlags | ((long) 2 & 0x0ffffffffL); 47 final long fieldId3 = fieldFlags | ((long) 3 & 0x0ffffffffL); 48 final long fieldId4 = fieldFlags | ((long) 4 & 0x0ffffffffL); 49 final long fieldId5 = fieldFlags | ((long) 5 & 0x0ffffffffL); 50 final long fieldId6 = fieldFlags | ((long) 6 & 0x0ffffffffL); 51 final long fieldId7 = fieldFlags | ((long) 7 & 0x0ffffffffL); 52 final long fieldId8 = fieldFlags | ((long) 8 & 0x0ffffffffL); 53 final long fieldId9 = fieldFlags | ((long) 9 & 0x0ffffffffL); 54 final long fieldId10 = fieldFlags | ((long) 10 & 0x0ffffffffL); 55 56 final byte[] protobuf = new byte[]{ 57 // 1 -> 0 - default value, not written 58 // 2 -> 1 59 (byte) 0x11, 60 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 61 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x3f, 62 // 10 -> 1 63 (byte) 0x51, 64 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 65 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x3f, 66 // 3 -> -1234.432 67 (byte) 0x19, 68 (byte) 0x7d, (byte) 0x3f, (byte) 0x35, (byte) 0x5e, 69 (byte) 0xba, (byte) 0x49, (byte) 0x93, (byte) 0xc0, 70 // 4 -> 42.42 71 (byte) 0x21, 72 (byte) 0xf6, (byte) 0x28, (byte) 0x5c, (byte) 0x8f, 73 (byte) 0xc2, (byte) 0x35, (byte) 0x45, (byte) 0x40, 74 // 5 -> Double.MIN_NORMAL 75 (byte) 0x29, 76 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 77 (byte) 0x00, (byte) 0x00, (byte) 0x10, (byte) 0x00, 78 // 6 -> DOUBLE.MIN_VALUE 79 (byte) 0x31, 80 (byte) 0x01, (byte) 0x00, (byte) 0x00, (byte) 0x00, 81 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 82 // 7 -> Double.NEGATIVE_INFINITY 83 (byte) 0x39, 84 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 85 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0xff, 86 // 8 -> Double.NaN 87 (byte) 0x41, 88 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 89 (byte) 0x00, (byte) 0x00, (byte) 0xf8, (byte) 0x7f, 90 // 9 -> Double.POSITIVE_INFINITY 91 (byte) 0x49, 92 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 93 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x7f, 94 }; 95 96 InputStream stream = new ByteArrayInputStream(protobuf); 97 final ProtoInputStream pi = new ProtoInputStream(stream, chunkSize); 98 double[] results = new double[9]; 99 while (pi.nextField() != ProtoInputStream.NO_MORE_FIELDS) { 100 switch (pi.getFieldNumber()) { 101 case (int) fieldId1: 102 fail("Should never reach this"); 103 break; 104 case (int) fieldId2: 105 results[1] = pi.readDouble(fieldId2); 106 break; 107 case (int) fieldId3: 108 results[2] = pi.readDouble(fieldId3); 109 break; 110 case (int) fieldId4: 111 results[3] = pi.readDouble(fieldId4); 112 break; 113 case (int) fieldId5: 114 results[4] = pi.readDouble(fieldId5); 115 break; 116 case (int) fieldId6: 117 results[5] = pi.readDouble(fieldId6); 118 break; 119 case (int) fieldId7: 120 results[6] = pi.readDouble(fieldId7); 121 break; 122 case (int) fieldId8: 123 results[7] = pi.readDouble(fieldId8); 124 break; 125 case (int) fieldId9: 126 results[8] = pi.readDouble(fieldId9); 127 break; 128 case (int) fieldId10: 129 // Intentionally don't read the data. Parse should continue normally 130 break; 131 default: 132 fail("Unexpected field id " + pi.getFieldNumber()); 133 } 134 } 135 stream.close(); 136 assertEquals(0.0, results[0]); 137 assertEquals(1.0, results[1]); 138 assertEquals(-1234.432, results[2]); 139 assertEquals(42.42, results[3]); 140 assertEquals(Double.MIN_NORMAL, results[4]); 141 assertEquals(Double.MIN_VALUE, results[5]); 142 assertEquals(Double.NEGATIVE_INFINITY, results[6]); 143 assertEquals(Double.NaN, results[7]); 144 assertEquals(Double.POSITIVE_INFINITY, results[8]); 145 } 146 147 /** 148 * Test that reading with ProtoInputStream matches, and can read the output of standard proto. 149 */ testReadCompat()150 public void testReadCompat() throws Exception { 151 testReadCompat(0); 152 testReadCompat(1); 153 testReadCompat(-1234.432); 154 testReadCompat(42.42); 155 testReadCompat(Double.MIN_NORMAL); 156 testReadCompat(Double.MIN_VALUE); 157 testReadCompat(Double.NEGATIVE_INFINITY); 158 testReadCompat(Double.NaN); 159 testReadCompat(Double.POSITIVE_INFINITY); 160 } 161 162 /** 163 * Implementation of testReadCompat with a given value. 164 */ testReadCompat(double val)165 private void testReadCompat(double val) throws Exception { 166 final long fieldFlags = ProtoStream.FIELD_COUNT_SINGLE | ProtoStream.FIELD_TYPE_DOUBLE; 167 final long fieldId = fieldFlags | ((long) 10 & 0x0ffffffffL); 168 169 final Test.All all = new Test.All(); 170 all.doubleField = val; 171 172 final byte[] proto = MessageNano.toByteArray(all); 173 174 final ProtoInputStream pi = new ProtoInputStream(proto); 175 final Test.All readback = Test.All.parseFrom(proto); 176 177 double result = 0.0; // start off with default value 178 while (pi.nextField() != ProtoInputStream.NO_MORE_FIELDS) { 179 switch (pi.getFieldNumber()) { 180 case (int) fieldId: 181 result = pi.readDouble(fieldId); 182 break; 183 default: 184 fail("Unexpected field id " + pi.getFieldNumber()); 185 } 186 } 187 188 assertEquals(readback.doubleField, result); 189 } 190 191 testRepeated()192 public void testRepeated() throws IOException { 193 testRepeated(0); 194 testRepeated(1); 195 testRepeated(5); 196 } 197 testRepeated(int chunkSize)198 private void testRepeated(int chunkSize) throws IOException { 199 final long fieldFlags = ProtoStream.FIELD_COUNT_REPEATED | ProtoStream.FIELD_TYPE_DOUBLE; 200 201 final long fieldId1 = fieldFlags | ((long) 1 & 0x0ffffffffL); 202 final long fieldId2 = fieldFlags | ((long) 2 & 0x0ffffffffL); 203 final long fieldId3 = fieldFlags | ((long) 3 & 0x0ffffffffL); 204 final long fieldId4 = fieldFlags | ((long) 4 & 0x0ffffffffL); 205 final long fieldId5 = fieldFlags | ((long) 5 & 0x0ffffffffL); 206 final long fieldId6 = fieldFlags | ((long) 6 & 0x0ffffffffL); 207 final long fieldId7 = fieldFlags | ((long) 7 & 0x0ffffffffL); 208 final long fieldId8 = fieldFlags | ((long) 8 & 0x0ffffffffL); 209 final long fieldId9 = fieldFlags | ((long) 9 & 0x0ffffffffL); 210 final long fieldId10 = fieldFlags | ((long) 10 & 0x0ffffffffL); 211 212 final byte[] protobuf = new byte[]{ 213 // 1 -> 0 - default value, written when repeated 214 (byte) 0x09, 215 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 216 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 217 // 2 -> 1 218 (byte) 0x11, 219 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 220 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x3f, 221 // 3 -> -1234.432 222 (byte) 0x19, 223 (byte) 0x7d, (byte) 0x3f, (byte) 0x35, (byte) 0x5e, 224 (byte) 0xba, (byte) 0x49, (byte) 0x93, (byte) 0xc0, 225 // 4 -> 42.42 226 (byte) 0x21, 227 (byte) 0xf6, (byte) 0x28, (byte) 0x5c, (byte) 0x8f, 228 (byte) 0xc2, (byte) 0x35, (byte) 0x45, (byte) 0x40, 229 // 5 -> Double.MIN_NORMAL 230 (byte) 0x29, 231 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 232 (byte) 0x00, (byte) 0x00, (byte) 0x10, (byte) 0x00, 233 // 6 -> DOUBLE.MIN_VALUE 234 (byte) 0x31, 235 (byte) 0x01, (byte) 0x00, (byte) 0x00, (byte) 0x00, 236 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 237 // 7 -> Double.NEGATIVE_INFINITY 238 (byte) 0x39, 239 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 240 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0xff, 241 // 8 -> Double.NaN 242 (byte) 0x41, 243 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 244 (byte) 0x00, (byte) 0x00, (byte) 0xf8, (byte) 0x7f, 245 // 9 -> Double.POSITIVE_INFINITY 246 (byte) 0x49, 247 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 248 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x7f, 249 // 10 -> 1 250 (byte) 0x51, 251 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 252 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x3f, 253 254 // 1 -> 0 - default value, written when repeated 255 (byte) 0x09, 256 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 257 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 258 // 2 -> 1 259 (byte) 0x11, 260 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 261 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x3f, 262 // 3 -> -1234.432 263 (byte) 0x19, 264 (byte) 0x7d, (byte) 0x3f, (byte) 0x35, (byte) 0x5e, 265 (byte) 0xba, (byte) 0x49, (byte) 0x93, (byte) 0xc0, 266 // 4 -> 42.42 267 (byte) 0x21, 268 (byte) 0xf6, (byte) 0x28, (byte) 0x5c, (byte) 0x8f, 269 (byte) 0xc2, (byte) 0x35, (byte) 0x45, (byte) 0x40, 270 // 5 -> Double.MIN_NORMAL 271 (byte) 0x29, 272 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 273 (byte) 0x00, (byte) 0x00, (byte) 0x10, (byte) 0x00, 274 // 6 -> DOUBLE.MIN_VALUE 275 (byte) 0x31, 276 (byte) 0x01, (byte) 0x00, (byte) 0x00, (byte) 0x00, 277 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 278 // 7 -> Double.NEGATIVE_INFINITY 279 (byte) 0x39, 280 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 281 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0xff, 282 // 8 -> Double.NaN 283 (byte) 0x41, 284 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 285 (byte) 0x00, (byte) 0x00, (byte) 0xf8, (byte) 0x7f, 286 // 9 -> Double.POSITIVE_INFINITY 287 (byte) 0x49, 288 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 289 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x7f, 290 }; 291 292 InputStream stream = new ByteArrayInputStream(protobuf); 293 final ProtoInputStream pi = new ProtoInputStream(stream, chunkSize); 294 double[][] results = new double[9][2]; 295 int[] indices = new int[9]; 296 while (pi.nextField() != ProtoInputStream.NO_MORE_FIELDS) { 297 298 switch (pi.getFieldNumber()) { 299 case (int) fieldId1: 300 results[0][indices[0]++] = pi.readDouble(fieldId1); 301 break; 302 case (int) fieldId2: 303 results[1][indices[1]++] = pi.readDouble(fieldId2); 304 break; 305 case (int) fieldId3: 306 results[2][indices[2]++] = pi.readDouble(fieldId3); 307 break; 308 case (int) fieldId4: 309 results[3][indices[3]++] = pi.readDouble(fieldId4); 310 break; 311 case (int) fieldId5: 312 results[4][indices[4]++] = pi.readDouble(fieldId5); 313 break; 314 case (int) fieldId6: 315 results[5][indices[5]++] = pi.readDouble(fieldId6); 316 break; 317 case (int) fieldId7: 318 results[6][indices[6]++] = pi.readDouble(fieldId7); 319 break; 320 case (int) fieldId8: 321 results[7][indices[7]++] = pi.readDouble(fieldId8); 322 break; 323 case (int) fieldId9: 324 results[8][indices[8]++] = pi.readDouble(fieldId9); 325 break; 326 case (int) fieldId10: 327 // Intentionally don't read the data. Parse should continue normally 328 break; 329 default: 330 fail("Unexpected field id " + pi.getFieldNumber()); 331 } 332 } 333 stream.close(); 334 assertEquals(0.0, results[0][0]); 335 assertEquals(0.0, results[0][1]); 336 assertEquals(1.0, results[1][0]); 337 assertEquals(1.0, results[1][1]); 338 assertEquals(-1234.432, results[2][0]); 339 assertEquals(-1234.432, results[2][1]); 340 assertEquals(42.42, results[3][0]); 341 assertEquals(42.42, results[3][1]); 342 assertEquals(Double.MIN_NORMAL, results[4][0]); 343 assertEquals(Double.MIN_NORMAL, results[4][1]); 344 assertEquals(Double.MIN_VALUE, results[5][0]); 345 assertEquals(Double.MIN_VALUE, results[5][1]); 346 assertEquals(Double.NEGATIVE_INFINITY, results[6][0]); 347 assertEquals(Double.NEGATIVE_INFINITY, results[6][1]); 348 assertEquals(Double.NaN, results[7][0]); 349 assertEquals(Double.NaN, results[7][1]); 350 assertEquals(Double.POSITIVE_INFINITY, results[8][0]); 351 assertEquals(Double.POSITIVE_INFINITY, results[8][1]); 352 } 353 354 /** 355 * Test that reading with ProtoInputStream matches, and can read the output of standard proto. 356 */ testRepeatedCompat()357 public void testRepeatedCompat() throws Exception { 358 testRepeatedCompat(new double[0]); 359 testRepeatedCompat(new double[]{0, 1, -1234.432, 42.42, 360 Double.MIN_NORMAL, Double.MIN_VALUE, Double.NEGATIVE_INFINITY, Double.NaN, 361 Double.POSITIVE_INFINITY, 362 }); 363 } 364 365 /** 366 * Implementation of testRepeatedCompat with a given value. 367 */ testRepeatedCompat(double[] val)368 private void testRepeatedCompat(double[] val) throws Exception { 369 final long fieldFlags = ProtoStream.FIELD_COUNT_REPEATED | ProtoStream.FIELD_TYPE_DOUBLE; 370 final long fieldId = fieldFlags | ((long) 11 & 0x0ffffffffL); 371 372 final Test.All all = new Test.All(); 373 all.doubleFieldRepeated = val; 374 375 final byte[] proto = MessageNano.toByteArray(all); 376 377 final ProtoInputStream pi = new ProtoInputStream(proto); 378 final Test.All readback = Test.All.parseFrom(proto); 379 380 double[] result = new double[val.length]; 381 int index = 0; 382 while (pi.nextField() != ProtoInputStream.NO_MORE_FIELDS) { 383 switch (pi.getFieldNumber()) { 384 case (int) fieldId: 385 result[index++] = pi.readDouble(fieldId); 386 break; 387 default: 388 fail("Unexpected field id " + pi.getFieldNumber()); 389 } 390 } 391 392 assertEquals(readback.doubleFieldRepeated.length, result.length); 393 for (int i = 0; i < result.length; i++) { 394 assertEquals(readback.doubleFieldRepeated[i], result[i]); 395 } 396 } 397 398 testPacked()399 public void testPacked() throws IOException { 400 testPacked(0); 401 testPacked(1); 402 testPacked(5); 403 } 404 testPacked(int chunkSize)405 private void testPacked(int chunkSize) throws IOException { 406 final long fieldFlags = ProtoStream.FIELD_COUNT_PACKED | ProtoStream.FIELD_TYPE_DOUBLE; 407 408 final long fieldId1 = fieldFlags | ((long) 1 & 0x0ffffffffL); 409 final long fieldId2 = fieldFlags | ((long) 2 & 0x0ffffffffL); 410 final long fieldId3 = fieldFlags | ((long) 3 & 0x0ffffffffL); 411 final long fieldId4 = fieldFlags | ((long) 4 & 0x0ffffffffL); 412 final long fieldId5 = fieldFlags | ((long) 5 & 0x0ffffffffL); 413 final long fieldId6 = fieldFlags | ((long) 6 & 0x0ffffffffL); 414 final long fieldId7 = fieldFlags | ((long) 7 & 0x0ffffffffL); 415 final long fieldId8 = fieldFlags | ((long) 8 & 0x0ffffffffL); 416 final long fieldId9 = fieldFlags | ((long) 9 & 0x0ffffffffL); 417 final long fieldId10 = fieldFlags | ((long) 10 & 0x0ffffffffL); 418 419 final byte[] protobuf = new byte[]{ 420 // 1 -> 0 - default value, written when repeated 421 (byte) 0x0a, 422 (byte) 0x10, 423 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 424 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 425 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 426 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 427 // 2 -> 1 428 (byte) 0x12, 429 (byte) 0x10, 430 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 431 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x3f, 432 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 433 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x3f, 434 // 10 -> 1 435 (byte) 0x52, 436 (byte) 0x10, 437 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 438 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x3f, 439 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 440 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x3f, 441 // 3 -> -1234.432 442 (byte) 0x1a, 443 (byte) 0x10, 444 (byte) 0x7d, (byte) 0x3f, (byte) 0x35, (byte) 0x5e, 445 (byte) 0xba, (byte) 0x49, (byte) 0x93, (byte) 0xc0, 446 (byte) 0x7d, (byte) 0x3f, (byte) 0x35, (byte) 0x5e, 447 (byte) 0xba, (byte) 0x49, (byte) 0x93, (byte) 0xc0, 448 // 4 -> 42.42 449 (byte) 0x22, 450 (byte) 0x10, 451 (byte) 0xf6, (byte) 0x28, (byte) 0x5c, (byte) 0x8f, 452 (byte) 0xc2, (byte) 0x35, (byte) 0x45, (byte) 0x40, 453 (byte) 0xf6, (byte) 0x28, (byte) 0x5c, (byte) 0x8f, 454 (byte) 0xc2, (byte) 0x35, (byte) 0x45, (byte) 0x40, 455 // 5 -> Double.MIN_NORMAL 456 (byte) 0x2a, 457 (byte) 0x10, 458 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 459 (byte) 0x00, (byte) 0x00, (byte) 0x10, (byte) 0x00, 460 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 461 (byte) 0x00, (byte) 0x00, (byte) 0x10, (byte) 0x00, 462 // 6 -> DOUBLE.MIN_VALUE 463 (byte) 0x32, 464 (byte) 0x10, 465 (byte) 0x01, (byte) 0x00, (byte) 0x00, (byte) 0x00, 466 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 467 (byte) 0x01, (byte) 0x00, (byte) 0x00, (byte) 0x00, 468 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 469 // 7 -> Double.NEGATIVE_INFINITY 470 (byte) 0x3a, 471 (byte) 0x10, 472 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 473 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0xff, 474 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 475 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0xff, 476 // 8 -> Double.NaN 477 (byte) 0x42, 478 (byte) 0x10, 479 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 480 (byte) 0x00, (byte) 0x00, (byte) 0xf8, (byte) 0x7f, 481 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 482 (byte) 0x00, (byte) 0x00, (byte) 0xf8, (byte) 0x7f, 483 // 9 -> Double.POSITIVE_INFINITY 484 (byte) 0x4a, 485 (byte) 0x10, 486 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 487 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x7f, 488 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 489 (byte) 0x00, (byte) 0x00, (byte) 0xf0, (byte) 0x7f, 490 }; 491 492 InputStream stream = new ByteArrayInputStream(protobuf); 493 final ProtoInputStream pi = new ProtoInputStream(stream, chunkSize); 494 double[][] results = new double[9][2]; 495 int[] indices = new int[9]; 496 while (pi.nextField() != ProtoInputStream.NO_MORE_FIELDS) { 497 498 switch (pi.getFieldNumber()) { 499 case (int) fieldId1: 500 results[0][indices[0]++] = pi.readDouble(fieldId1); 501 break; 502 case (int) fieldId2: 503 results[1][indices[1]++] = pi.readDouble(fieldId2); 504 break; 505 case (int) fieldId3: 506 results[2][indices[2]++] = pi.readDouble(fieldId3); 507 break; 508 case (int) fieldId4: 509 results[3][indices[3]++] = pi.readDouble(fieldId4); 510 break; 511 case (int) fieldId5: 512 results[4][indices[4]++] = pi.readDouble(fieldId5); 513 break; 514 case (int) fieldId6: 515 results[5][indices[5]++] = pi.readDouble(fieldId6); 516 break; 517 case (int) fieldId7: 518 results[6][indices[6]++] = pi.readDouble(fieldId7); 519 break; 520 case (int) fieldId8: 521 results[7][indices[7]++] = pi.readDouble(fieldId8); 522 break; 523 case (int) fieldId9: 524 results[8][indices[8]++] = pi.readDouble(fieldId9); 525 break; 526 case (int) fieldId10: 527 // Intentionally don't read the data. Parse should continue normally 528 break; 529 default: 530 fail("Unexpected field id " + pi.getFieldNumber()); 531 } 532 } 533 stream.close(); 534 assertEquals(0.0, results[0][0]); 535 assertEquals(0.0, results[0][1]); 536 assertEquals(1.0, results[1][0]); 537 assertEquals(1.0, results[1][1]); 538 assertEquals(-1234.432, results[2][0]); 539 assertEquals(-1234.432, results[2][1]); 540 assertEquals(42.42, results[3][0]); 541 assertEquals(42.42, results[3][1]); 542 assertEquals(Double.MIN_NORMAL, results[4][0]); 543 assertEquals(Double.MIN_NORMAL, results[4][1]); 544 assertEquals(Double.MIN_VALUE, results[5][0]); 545 assertEquals(Double.MIN_VALUE, results[5][1]); 546 assertEquals(Double.NEGATIVE_INFINITY, results[6][0]); 547 assertEquals(Double.NEGATIVE_INFINITY, results[6][1]); 548 assertEquals(Double.NaN, results[7][0]); 549 assertEquals(Double.NaN, results[7][1]); 550 assertEquals(Double.POSITIVE_INFINITY, results[8][0]); 551 assertEquals(Double.POSITIVE_INFINITY, results[8][1]); 552 } 553 554 /** 555 * Test that reading with ProtoInputStream matches, and can read the output of standard proto. 556 */ testPackedCompat()557 public void testPackedCompat() throws Exception { 558 testPackedCompat(new double[0]); 559 testPackedCompat(new double[]{0, 1, -1234.432, 42.42, 560 Double.MIN_NORMAL, Double.MIN_VALUE, Double.NEGATIVE_INFINITY, Double.NaN, 561 Double.POSITIVE_INFINITY, 562 }); 563 } 564 565 /** 566 * Implementation of testPackedCompat with a given value. 567 */ testPackedCompat(double[] val)568 private void testPackedCompat(double[] val) throws Exception { 569 final long fieldFlags = ProtoStream.FIELD_COUNT_PACKED | ProtoStream.FIELD_TYPE_DOUBLE; 570 final long fieldId = fieldFlags | ((long) 12 & 0x0ffffffffL); 571 572 final Test.All all = new Test.All(); 573 all.doubleFieldPacked = val; 574 575 final byte[] proto = MessageNano.toByteArray(all); 576 577 final ProtoInputStream pi = new ProtoInputStream(proto); 578 final Test.All readback = Test.All.parseFrom(proto); 579 580 double[] result = new double[val.length]; 581 int index = 0; 582 while (pi.nextField() != ProtoInputStream.NO_MORE_FIELDS) { 583 switch (pi.getFieldNumber()) { 584 case (int) fieldId: 585 result[index++] = pi.readDouble(fieldId); 586 break; 587 default: 588 fail("Unexpected field id " + pi.getFieldNumber()); 589 } 590 } 591 592 assertEquals(readback.doubleFieldPacked.length, result.length); 593 for (int i = 0; i < result.length; i++) { 594 assertEquals(readback.doubleFieldPacked[i], result[i]); 595 } 596 } 597 598 /** 599 * Test that using the wrong read method throws an exception 600 */ testBadReadType()601 public void testBadReadType() throws IOException { 602 final long fieldFlags = ProtoStream.FIELD_COUNT_SINGLE | ProtoStream.FIELD_TYPE_DOUBLE; 603 604 final long fieldId1 = fieldFlags | ((long) 1 & 0x0ffffffffL); 605 606 final byte[] protobuf = new byte[]{ 607 // 1 -> 1 608 (byte) 0x09, 609 (byte) 0x01, (byte) 0x00, (byte) 0x00, (byte) 0x00, 610 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 611 }; 612 613 ProtoInputStream pi = new ProtoInputStream(protobuf); 614 pi.isNextField(fieldId1); 615 try { 616 pi.readFloat(fieldId1); 617 fail("Should have throw IllegalArgumentException"); 618 } catch (IllegalArgumentException iae) { 619 // good 620 } 621 622 pi = new ProtoInputStream(protobuf); 623 pi.isNextField(fieldId1); 624 try { 625 pi.readBoolean(fieldId1); 626 fail("Should have throw IllegalArgumentException"); 627 } catch (IllegalArgumentException iae) { 628 // good 629 } 630 631 pi = new ProtoInputStream(protobuf); 632 pi.isNextField(fieldId1); 633 try { 634 pi.readInt(fieldId1); 635 fail("Should have throw IllegalArgumentException"); 636 } catch (IllegalArgumentException iae) { 637 // good 638 } 639 640 pi = new ProtoInputStream(protobuf); 641 pi.isNextField(fieldId1); 642 try { 643 pi.readLong(fieldId1); 644 fail("Should have throw IllegalArgumentException"); 645 } catch (IllegalArgumentException iae) { 646 // good 647 } 648 649 pi = new ProtoInputStream(protobuf); 650 pi.isNextField(fieldId1); 651 try { 652 pi.readBytes(fieldId1); 653 fail("Should have throw IllegalArgumentException"); 654 } catch (IllegalArgumentException iae) { 655 // good 656 } 657 658 pi = new ProtoInputStream(protobuf); 659 pi.isNextField(fieldId1); 660 try { 661 pi.readString(fieldId1); 662 fail("Should have throw IllegalArgumentException"); 663 } catch (IllegalArgumentException iae) { 664 // good 665 } 666 } 667 668 /** 669 * Test that unexpected wrong wire types will throw an exception 670 */ testBadWireType()671 public void testBadWireType() throws IOException { 672 final long fieldFlags = ProtoStream.FIELD_COUNT_SINGLE | ProtoStream.FIELD_TYPE_DOUBLE; 673 674 final long fieldId1 = fieldFlags | ((long) 1 & 0x0ffffffffL); 675 final long fieldId2 = fieldFlags | ((long) 2 & 0x0ffffffffL); 676 final long fieldId3 = fieldFlags | ((long) 3 & 0x0ffffffffL); 677 final long fieldId6 = fieldFlags | ((long) 6 & 0x0ffffffffL); 678 679 final byte[] protobuf = new byte[]{ 680 // 1 : varint -> 1 681 (byte) 0x08, 682 (byte) 0x01, 683 // 2 : fixed64 -> 0x1 684 (byte) 0x11, 685 (byte) 0x01, (byte) 0x00, (byte) 0x00, (byte) 0x00, 686 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 687 // 3 : length delimited -> { 1 } 688 (byte) 0x1a, 689 (byte) 0x08, 690 (byte) 0x01, (byte) 0x00, (byte) 0x00, (byte) 0x00, 691 (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, 692 // 6 : fixed32 693 (byte) 0x35, 694 (byte) 0x01, (byte) 0x00, (byte) 0x00, (byte) 0x00, 695 }; 696 697 InputStream stream = new ByteArrayInputStream(protobuf); 698 final ProtoInputStream pi = new ProtoInputStream(stream); 699 700 while (pi.nextField() != ProtoInputStream.NO_MORE_FIELDS) { 701 try { 702 switch (pi.getFieldNumber()) { 703 case (int) fieldId1: 704 pi.readDouble(fieldId1); 705 fail("Should have thrown a WireTypeMismatchException"); 706 break; 707 case (int) fieldId2: 708 pi.readDouble(fieldId2); 709 // don't fail, fixed64 is ok 710 break; 711 case (int) fieldId3: 712 pi.readDouble(fieldId3); 713 // don't fail, length delimited is ok (represents packed doubles) 714 break; 715 case (int) fieldId6: 716 pi.readDouble(fieldId6); 717 fail("Should have thrown a WireTypeMismatchException"); 718 break; 719 default: 720 fail("Unexpected field id " + pi.getFieldNumber()); 721 } 722 } catch (WireTypeMismatchException wtme) { 723 // good 724 } 725 } 726 stream.close(); 727 } 728 } 729