001// Copyright (c) FIRST and other WPILib contributors. 002// Open Source Software; you can modify and/or share it under the terms of 003// the WPILib BSD license file in the root directory of this project. 004 005package edu.wpi.first.util.struct; 006 007import java.nio.BufferUnderflowException; 008import java.nio.ByteBuffer; 009import java.nio.ByteOrder; 010import java.nio.ReadOnlyBufferException; 011import java.nio.charset.StandardCharsets; 012 013/** Dynamic (run-time) access to a serialized raw struct. */ 014public final class DynamicStruct { 015 private DynamicStruct(StructDescriptor desc, ByteBuffer data) { 016 m_desc = desc; 017 m_data = data.order(ByteOrder.LITTLE_ENDIAN); 018 } 019 020 /** 021 * Constructs a new dynamic struct object with internal storage. The descriptor must be valid. The 022 * internal storage is allocated using ByteBuffer.allocate(). 023 * 024 * @param desc struct descriptor 025 * @return dynamic struct object 026 * @throws IllegalStateException if struct descriptor is invalid 027 */ 028 public static DynamicStruct allocate(StructDescriptor desc) { 029 return new DynamicStruct(desc, ByteBuffer.allocate(desc.getSize())); 030 } 031 032 /** 033 * Constructs a new dynamic struct object with internal storage. The descriptor must be valid. The 034 * internal storage is allocated using ByteBuffer.allocateDirect(). 035 * 036 * @param desc struct descriptor 037 * @return dynamic struct object 038 * @throws IllegalStateException if struct descriptor is invalid 039 */ 040 public static DynamicStruct allocateDirect(StructDescriptor desc) { 041 return new DynamicStruct(desc, ByteBuffer.allocateDirect(desc.getSize())); 042 } 043 044 /** 045 * Constructs a new dynamic struct object. Note: the passed data buffer is not copied. 046 * Modifications to the passed buffer will be reflected in the struct and vice-versa. 047 * 048 * @param desc struct descriptor 049 * @param data byte buffer containing serialized data starting at current position 050 * @return dynamic struct object 051 */ 052 public static DynamicStruct wrap(StructDescriptor desc, ByteBuffer data) { 053 return new DynamicStruct(desc, data.slice()); 054 } 055 056 /** 057 * Gets the struct descriptor. 058 * 059 * @return struct descriptor 060 */ 061 public StructDescriptor getDescriptor() { 062 return m_desc; 063 } 064 065 /** 066 * Gets the serialized backing data buffer. 067 * 068 * @return data buffer 069 */ 070 public ByteBuffer getBuffer() { 071 return m_data.duplicate().position(0); 072 } 073 074 /** 075 * Overwrites the entire serialized struct by copying data from a byte array. 076 * 077 * @param data replacement data for the struct 078 * @throws BufferUnderflowException if data is smaller than the struct size 079 * @throws ReadOnlyBufferException if the underlying buffer is read-only 080 * @throws IllegalStateException if struct descriptor is invalid 081 */ 082 public void setData(byte[] data) { 083 if (data.length < m_desc.getSize()) { 084 throw new BufferUnderflowException(); 085 } 086 m_data.position(0).put(data); 087 } 088 089 /** 090 * Overwrites the entire serialized struct by copying data from a byte buffer. 091 * 092 * @param data replacement data for the struct; copy starts from current position 093 * @throws BufferUnderflowException if remaining data is smaller than the struct size 094 * @throws ReadOnlyBufferException if the underlying buffer is read-only 095 * @throws IllegalStateException if struct descriptor is invalid 096 */ 097 public void setData(ByteBuffer data) { 098 if (data.remaining() < m_desc.getSize()) { 099 throw new BufferUnderflowException(); 100 } 101 int oldLimit = data.limit(); 102 m_data.position(0).put(data.limit(m_desc.getSize())); 103 data.limit(oldLimit); 104 } 105 106 /** 107 * Gets a struct field descriptor by name. 108 * 109 * @param name field name 110 * @return field descriptor, or null if no field with that name exists 111 */ 112 public StructFieldDescriptor findField(String name) { 113 return m_desc.findFieldByName(name); 114 } 115 116 /** 117 * Gets the value of a boolean field. 118 * 119 * @param field field descriptor 120 * @param arrIndex array index (must be less than field array size) 121 * @return boolean field value 122 * @throws UnsupportedOperationException if field is not bool type 123 * @throws IllegalArgumentException if field is not a member of this struct 124 * @throws IllegalStateException if struct descriptor is invalid 125 * @throws ArrayIndexOutOfBoundsException if array index is out of bounds 126 */ 127 public boolean getBoolField(StructFieldDescriptor field, int arrIndex) { 128 if (field.getType() != StructFieldType.kBool) { 129 throw new UnsupportedOperationException("field is not bool type"); 130 } 131 return getFieldImpl(field, arrIndex) != 0; 132 } 133 134 /** 135 * Gets the value of a boolean field. 136 * 137 * @param field field descriptor 138 * @return boolean field value 139 * @throws UnsupportedOperationException if field is not bool type 140 * @throws IllegalArgumentException if field is not a member of this struct 141 * @throws IllegalStateException if struct descriptor is invalid 142 */ 143 public boolean getBoolField(StructFieldDescriptor field) { 144 return getBoolField(field, 0); 145 } 146 147 /** 148 * Sets the value of a boolean field. 149 * 150 * @param field field descriptor 151 * @param value boolean value 152 * @param arrIndex array index (must be less than field array size) 153 * @throws UnsupportedOperationException if field is not bool type 154 * @throws IllegalArgumentException if field is not a member of this struct 155 * @throws IllegalStateException if struct descriptor is invalid 156 * @throws ArrayIndexOutOfBoundsException if array index is out of bounds 157 * @throws ReadOnlyBufferException if the underlying buffer is read-only 158 */ 159 public void setBoolField(StructFieldDescriptor field, boolean value, int arrIndex) { 160 if (field.getType() != StructFieldType.kBool) { 161 throw new UnsupportedOperationException("field is not bool type"); 162 } 163 setFieldImpl(field, value ? 1 : 0, arrIndex); 164 } 165 166 /** 167 * Sets the value of a boolean field. 168 * 169 * @param field field descriptor 170 * @param value boolean value 171 * @throws UnsupportedOperationException if field is not bool type 172 * @throws IllegalArgumentException if field is not a member of this struct 173 * @throws IllegalStateException if struct descriptor is invalid 174 * @throws ReadOnlyBufferException if the underlying buffer is read-only 175 */ 176 public void setBoolField(StructFieldDescriptor field, boolean value) { 177 setBoolField(field, value, 0); 178 } 179 180 /** 181 * Gets the value of an integer field. 182 * 183 * @param field field descriptor 184 * @param arrIndex array index (must be less than field array size) 185 * @return integer field value 186 * @throws UnsupportedOperationException if field is not integer type 187 * @throws IllegalArgumentException if field is not a member of this struct 188 * @throws IllegalStateException if struct descriptor is invalid 189 * @throws ArrayIndexOutOfBoundsException if array index is out of bounds 190 */ 191 public long getIntField(StructFieldDescriptor field, int arrIndex) { 192 if (!field.isInt() && !field.isUint()) { 193 throw new UnsupportedOperationException("field is not integer type"); 194 } 195 return getFieldImpl(field, arrIndex); 196 } 197 198 /** 199 * Gets the value of an integer field. 200 * 201 * @param field field descriptor 202 * @return integer field value 203 * @throws UnsupportedOperationException if field is not integer type 204 * @throws IllegalArgumentException if field is not a member of this struct 205 * @throws IllegalStateException if struct descriptor is invalid 206 */ 207 public long getIntField(StructFieldDescriptor field) { 208 return getIntField(field, 0); 209 } 210 211 /** 212 * Sets the value of an integer field. 213 * 214 * @param field field descriptor 215 * @param value integer value 216 * @param arrIndex array index (must be less than field array size) 217 * @throws UnsupportedOperationException if field is not integer type 218 * @throws IllegalArgumentException if field is not a member of this struct 219 * @throws IllegalStateException if struct descriptor is invalid 220 * @throws ArrayIndexOutOfBoundsException if array index is out of bounds 221 * @throws ReadOnlyBufferException if the underlying buffer is read-only 222 */ 223 public void setIntField(StructFieldDescriptor field, long value, int arrIndex) { 224 if (!field.isInt() && !field.isUint()) { 225 throw new UnsupportedOperationException("field is not integer type"); 226 } 227 setFieldImpl(field, value, arrIndex); 228 } 229 230 /** 231 * Sets the value of an integer field. 232 * 233 * @param field field descriptor 234 * @param value integer value 235 * @throws UnsupportedOperationException if field is not integer type 236 * @throws IllegalArgumentException if field is not a member of this struct 237 * @throws IllegalStateException if struct descriptor is invalid 238 * @throws ReadOnlyBufferException if the underlying buffer is read-only 239 */ 240 public void setIntField(StructFieldDescriptor field, long value) { 241 setIntField(field, value, 0); 242 } 243 244 /** 245 * Gets the value of a float field. 246 * 247 * @param field field descriptor 248 * @param arrIndex array index (must be less than field array size) 249 * @return float field value 250 * @throws UnsupportedOperationException if field is not float type 251 * @throws IllegalArgumentException if field is not a member of this struct 252 * @throws IllegalStateException if struct descriptor is invalid 253 * @throws ArrayIndexOutOfBoundsException if array index is out of bounds 254 */ 255 public float getFloatField(StructFieldDescriptor field, int arrIndex) { 256 if (field.getType() != StructFieldType.kFloat) { 257 throw new UnsupportedOperationException("field is not float type"); 258 } 259 return Float.intBitsToFloat((int) getFieldImpl(field, arrIndex)); 260 } 261 262 /** 263 * Gets the value of a float field. 264 * 265 * @param field field descriptor 266 * @return float field value 267 * @throws UnsupportedOperationException if field is not float type 268 * @throws IllegalArgumentException if field is not a member of this struct 269 * @throws IllegalStateException if struct descriptor is invalid 270 */ 271 public float getFloatField(StructFieldDescriptor field) { 272 return getFloatField(field, 0); 273 } 274 275 /** 276 * Sets the value of a float field. 277 * 278 * @param field field descriptor 279 * @param value float value 280 * @param arrIndex array index (must be less than field array size) 281 * @throws UnsupportedOperationException if field is not float type 282 * @throws IllegalArgumentException if field is not a member of this struct 283 * @throws IllegalStateException if struct descriptor is invalid 284 * @throws ArrayIndexOutOfBoundsException if array index is out of bounds 285 * @throws ReadOnlyBufferException if the underlying buffer is read-only 286 */ 287 public void setFloatField(StructFieldDescriptor field, float value, int arrIndex) { 288 if (field.getType() != StructFieldType.kFloat) { 289 throw new UnsupportedOperationException("field is not float type"); 290 } 291 setFieldImpl(field, Float.floatToIntBits(value), arrIndex); 292 } 293 294 /** 295 * Sets the value of a float field. 296 * 297 * @param field field descriptor 298 * @param value float value 299 * @throws UnsupportedOperationException if field is not float type 300 * @throws IllegalArgumentException if field is not a member of this struct 301 * @throws IllegalStateException if struct descriptor is invalid 302 * @throws ReadOnlyBufferException if the underlying buffer is read-only 303 */ 304 public void setFloatField(StructFieldDescriptor field, float value) { 305 setFloatField(field, value, 0); 306 } 307 308 /** 309 * Gets the value of a double field. 310 * 311 * @param field field descriptor 312 * @param arrIndex array index (must be less than field array size) 313 * @return double field value 314 * @throws UnsupportedOperationException if field is not double type 315 * @throws IllegalArgumentException if field is not a member of this struct 316 * @throws IllegalStateException if struct descriptor is invalid 317 * @throws ArrayIndexOutOfBoundsException if array index is out of bounds 318 */ 319 public double getDoubleField(StructFieldDescriptor field, int arrIndex) { 320 if (field.getType() != StructFieldType.kDouble) { 321 throw new UnsupportedOperationException("field is not double type"); 322 } 323 return Double.longBitsToDouble(getFieldImpl(field, arrIndex)); 324 } 325 326 /** 327 * Gets the value of a double field. 328 * 329 * @param field field descriptor 330 * @return double field value 331 * @throws UnsupportedOperationException if field is not double type 332 * @throws IllegalArgumentException if field is not a member of this struct 333 * @throws IllegalStateException if struct descriptor is invalid 334 */ 335 public double getDoubleField(StructFieldDescriptor field) { 336 return getDoubleField(field, 0); 337 } 338 339 /** 340 * Sets the value of a double field. 341 * 342 * @param field field descriptor 343 * @param value double value 344 * @param arrIndex array index (must be less than field array size) 345 * @throws UnsupportedOperationException if field is not double type 346 * @throws IllegalArgumentException if field is not a member of this struct 347 * @throws IllegalStateException if struct descriptor is invalid 348 * @throws ArrayIndexOutOfBoundsException if array index is out of bounds 349 * @throws ReadOnlyBufferException if the underlying buffer is read-only 350 */ 351 public void setDoubleField(StructFieldDescriptor field, double value, int arrIndex) { 352 if (field.getType() != StructFieldType.kDouble) { 353 throw new UnsupportedOperationException("field is not double type"); 354 } 355 setFieldImpl(field, Double.doubleToLongBits(value), arrIndex); 356 } 357 358 /** 359 * Sets the value of a double field. 360 * 361 * @param field field descriptor 362 * @param value double value 363 * @throws UnsupportedOperationException if field is not double type 364 * @throws IllegalArgumentException if field is not a member of this struct 365 * @throws IllegalStateException if struct descriptor is invalid 366 * @throws ReadOnlyBufferException if the underlying buffer is read-only 367 */ 368 public void setDoubleField(StructFieldDescriptor field, double value) { 369 setDoubleField(field, value, 0); 370 } 371 372 /** 373 * Gets the value of a character or character array field. 374 * 375 * @param field field descriptor 376 * @return field value 377 * @throws UnsupportedOperationException if field is not char type 378 * @throws IllegalArgumentException if field is not a member of this struct 379 * @throws IllegalStateException if struct descriptor is invalid 380 */ 381 @SuppressWarnings({"PMD.CollapsibleIfStatements", "PMD.AvoidDeeplyNestedIfStmts"}) 382 public String getStringField(StructFieldDescriptor field) { 383 if (field.getType() != StructFieldType.kChar) { 384 throw new UnsupportedOperationException("field is not char type"); 385 } 386 if (!field.getParent().equals(m_desc)) { 387 throw new IllegalArgumentException("field is not part of this struct"); 388 } 389 if (!m_desc.isValid()) { 390 throw new IllegalStateException("struct descriptor is not valid"); 391 } 392 byte[] bytes = new byte[field.m_arraySize]; 393 m_data.position(field.m_offset).get(bytes, 0, field.m_arraySize); 394 // Find last non zero character 395 int stringLength = bytes.length; 396 for (; stringLength > 0; stringLength--) { 397 if (bytes[stringLength - 1] != 0) { 398 break; 399 } 400 } 401 // If string is all zeroes, its empty and return an empty string. 402 if (stringLength == 0) { 403 return ""; 404 } 405 // Check if the end of the string is in the middle of a continuation byte or 406 // not. 407 if ((bytes[stringLength - 1] & 0x80) != 0) { 408 // This is a UTF8 continuation byte. Make sure its valid. 409 // Walk back until initial byte is found 410 int utf8StartByte = stringLength; 411 for (; utf8StartByte > 0; utf8StartByte--) { 412 if ((bytes[utf8StartByte - 1] & 0x40) != 0) { 413 // Having 2nd bit set means start byte 414 break; 415 } 416 } 417 if (utf8StartByte == 0) { 418 // This case means string only contains continuation bytes 419 return ""; 420 } 421 utf8StartByte--; 422 // Check if its a 2, 3, or 4 byte 423 byte checkByte = bytes[utf8StartByte]; 424 if ((checkByte & 0xE0) == 0xC0) { 425 // 2 byte, need 1 more byte 426 if (utf8StartByte != stringLength - 2) { 427 stringLength = utf8StartByte; 428 } 429 } else if ((checkByte & 0xF0) == 0xE0) { 430 // 3 byte, need 2 more bytes 431 if (utf8StartByte != stringLength - 3) { 432 stringLength = utf8StartByte; 433 } 434 } else if ((checkByte & 0xF8) == 0xF0) { 435 // 4 byte, need 3 more bytes 436 if (utf8StartByte != stringLength - 4) { 437 stringLength = utf8StartByte; 438 } 439 } 440 // If we get here, the string is either completely garbage or fine. 441 } 442 443 return new String(bytes, 0, stringLength, StandardCharsets.UTF_8); 444 } 445 446 /** 447 * Sets the value of a character or character array field. 448 * 449 * @param field field descriptor 450 * @param value field value 451 * @return true if the full value fit in the struct, false if truncated 452 * @throws UnsupportedOperationException if field is not char type 453 * @throws IllegalArgumentException if field is not a member of this struct 454 * @throws IllegalStateException if struct descriptor is invalid 455 */ 456 public boolean setStringField(StructFieldDescriptor field, String value) { 457 if (field.getType() != StructFieldType.kChar) { 458 throw new UnsupportedOperationException("field is not char type"); 459 } 460 if (!field.getParent().equals(m_desc)) { 461 throw new IllegalArgumentException("field is not part of this struct"); 462 } 463 if (!m_desc.isValid()) { 464 throw new IllegalStateException("struct descriptor is not valid"); 465 } 466 ByteBuffer bb = StandardCharsets.UTF_8.encode(value); 467 int len = Math.min(bb.remaining(), field.m_arraySize); 468 boolean copiedFull = len == bb.remaining(); 469 m_data.position(field.m_offset).put(bb.limit(len)); 470 for (int i = len; i < field.m_arraySize; i++) { 471 m_data.put((byte) 0); 472 } 473 return copiedFull; 474 } 475 476 /** 477 * Gets the value of a struct field. 478 * 479 * @param field field descriptor 480 * @param arrIndex array index (must be less than field array size) 481 * @return field value 482 * @throws UnsupportedOperationException if field is not of struct type 483 * @throws IllegalArgumentException if field is not a member of this struct 484 * @throws IllegalStateException if struct descriptor is invalid 485 * @throws ArrayIndexOutOfBoundsException if array index is out of bounds 486 */ 487 public DynamicStruct getStructField(StructFieldDescriptor field, int arrIndex) { 488 if (field.getType() != StructFieldType.kStruct) { 489 throw new UnsupportedOperationException("field is not struct type"); 490 } 491 if (!field.getParent().equals(m_desc)) { 492 throw new IllegalArgumentException("field is not part of this struct"); 493 } 494 if (!m_desc.isValid()) { 495 throw new IllegalStateException("struct descriptor is not valid"); 496 } 497 if (arrIndex < 0 || arrIndex >= field.m_arraySize) { 498 throw new ArrayIndexOutOfBoundsException( 499 "arrIndex (" + arrIndex + ") is larger than array size (" + field.m_arraySize + ")"); 500 } 501 StructDescriptor struct = field.getStruct(); 502 return wrap(struct, m_data.position(field.m_offset + arrIndex * struct.m_size)); 503 } 504 505 /** 506 * Gets the value of a struct field. 507 * 508 * @param field field descriptor 509 * @return field value 510 * @throws UnsupportedOperationException if field is not of struct type 511 * @throws IllegalArgumentException if field is not a member of this struct 512 * @throws IllegalStateException if struct descriptor is invalid 513 */ 514 public DynamicStruct getStructField(StructFieldDescriptor field) { 515 return getStructField(field, 0); 516 } 517 518 /** 519 * Sets the value of a struct field. 520 * 521 * @param field field descriptor 522 * @param value struct value 523 * @param arrIndex array index (must be less than field array size) 524 * @throws UnsupportedOperationException if field is not struct type 525 * @throws IllegalArgumentException if field is not a member of this struct 526 * @throws IllegalStateException if struct descriptor is invalid 527 * @throws ArrayIndexOutOfBoundsException if array index is out of bounds 528 * @throws ReadOnlyBufferException if the underlying buffer is read-only 529 */ 530 public void setStructField(StructFieldDescriptor field, DynamicStruct value, int arrIndex) { 531 if (field.getType() != StructFieldType.kStruct) { 532 throw new UnsupportedOperationException("field is not struct type"); 533 } 534 if (!field.getParent().equals(m_desc)) { 535 throw new IllegalArgumentException("field is not part of this struct"); 536 } 537 if (!m_desc.isValid()) { 538 throw new IllegalStateException("struct descriptor is not valid"); 539 } 540 StructDescriptor struct = field.getStruct(); 541 if (!value.getDescriptor().equals(struct)) { 542 throw new IllegalArgumentException("value's struct type does not match field struct type"); 543 } 544 if (!value.getDescriptor().isValid()) { 545 throw new IllegalStateException("value's struct descriptor is not valid"); 546 } 547 if (arrIndex < 0 || arrIndex >= field.m_arraySize) { 548 throw new ArrayIndexOutOfBoundsException( 549 "arrIndex (" + arrIndex + ") is larger than array size (" + field.m_arraySize + ")"); 550 } 551 m_data 552 .position(field.m_offset + arrIndex * struct.m_size) 553 .put(value.m_data.position(0).limit(value.getDescriptor().getSize())); 554 } 555 556 /** 557 * Sets the value of a struct field. 558 * 559 * @param field field descriptor 560 * @param value struct value 561 * @throws UnsupportedOperationException if field is not struct type 562 * @throws IllegalArgumentException if field is not a member of this struct 563 * @throws IllegalStateException if struct descriptor is invalid 564 * @throws ReadOnlyBufferException if the underlying buffer is read-only 565 */ 566 public void setStructField(StructFieldDescriptor field, DynamicStruct value) { 567 setStructField(field, value, 0); 568 } 569 570 private long getFieldImpl(StructFieldDescriptor field, int arrIndex) { 571 if (!field.getParent().equals(m_desc)) { 572 throw new IllegalArgumentException("field is not part of this struct"); 573 } 574 if (!m_desc.isValid()) { 575 throw new IllegalStateException("struct descriptor is not valid"); 576 } 577 if (arrIndex < 0 || arrIndex >= field.m_arraySize) { 578 throw new ArrayIndexOutOfBoundsException( 579 "arrIndex (" + arrIndex + ") is larger than array size (" + field.m_arraySize + ")"); 580 } 581 582 long val = 583 switch (field.m_size) { 584 case 1 -> m_data.get(field.m_offset + arrIndex); 585 case 2 -> m_data.getShort(field.m_offset + arrIndex * 2); 586 case 4 -> m_data.getInt(field.m_offset + arrIndex * 4); 587 case 8 -> m_data.getLong(field.m_offset + arrIndex * 8); 588 default -> throw new IllegalStateException("invalid field size"); 589 }; 590 591 if (field.isUint() || field.getType() == StructFieldType.kBool) { 592 // for unsigned fields, we can simply logical shift and mask 593 return (val >>> field.m_bitShift) & field.getBitMask(); 594 } else { 595 // to get sign extension, shift so the sign bit within the bitfield goes to the long's sign 596 // bit (also clearing all higher bits), then shift back down (also clearing all lower bits); 597 // since upper and lower bits are cleared with the shifts, the bitmask is unnecessary 598 return (val << (64 - field.m_bitShift - field.getBitWidth())) >> (64 - field.getBitWidth()); 599 } 600 } 601 602 private void setFieldImpl(StructFieldDescriptor field, long value, int arrIndex) { 603 if (!field.getParent().equals(m_desc)) { 604 throw new IllegalArgumentException("field is not part of this struct"); 605 } 606 if (!m_desc.isValid()) { 607 throw new IllegalStateException("struct descriptor is not valid"); 608 } 609 if (arrIndex < 0 || arrIndex >= field.m_arraySize) { 610 throw new ArrayIndexOutOfBoundsException( 611 "arrIndex (" + arrIndex + ") is larger than array size (" + field.m_arraySize + ")"); 612 } 613 614 // common case is no bit shift and no masking 615 if (!field.isBitField()) { 616 switch (field.m_size) { 617 case 1 -> m_data.put(field.m_offset + arrIndex, (byte) value); 618 case 2 -> m_data.putShort(field.m_offset + arrIndex * 2, (short) value); 619 case 4 -> m_data.putInt(field.m_offset + arrIndex * 4, (int) value); 620 case 8 -> m_data.putLong(field.m_offset + arrIndex * 8, value); 621 default -> throw new IllegalStateException("invalid field size"); 622 } 623 return; 624 } 625 626 // handle bit shifting and masking into current value 627 switch (field.m_size) { 628 case 1 -> { 629 byte val = m_data.get(field.m_offset + arrIndex); 630 val &= (byte) ~(field.getBitMask() << field.m_bitShift); 631 val |= (byte) ((value & field.getBitMask()) << field.m_bitShift); 632 m_data.put(field.m_offset + arrIndex, val); 633 } 634 case 2 -> { 635 short val = m_data.getShort(field.m_offset + arrIndex * 2); 636 val &= (short) ~(field.getBitMask() << field.m_bitShift); 637 val |= (short) ((value & field.getBitMask()) << field.m_bitShift); 638 m_data.putShort(field.m_offset + arrIndex * 2, val); 639 } 640 case 4 -> { 641 int val = m_data.getInt(field.m_offset + arrIndex * 4); 642 val &= (int) ~(field.getBitMask() << field.m_bitShift); 643 val |= (int) ((value & field.getBitMask()) << field.m_bitShift); 644 m_data.putInt(field.m_offset + arrIndex * 4, val); 645 } 646 case 8 -> { 647 long val = m_data.getLong(field.m_offset + arrIndex * 8); 648 val &= ~(field.getBitMask() << field.m_bitShift); 649 val |= (value & field.getBitMask()) << field.m_bitShift; 650 m_data.putLong(field.m_offset + arrIndex * 8, val); 651 } 652 default -> throw new IllegalStateException("invalid field size"); 653 } 654 } 655 656 private final StructDescriptor m_desc; 657 private final ByteBuffer m_data; 658}