2014-09-09 18:46:13 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2014 Google Inc. All rights reserved.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.Text;
|
2019-09-11 00:07:46 +00:00
|
|
|
using System.Runtime.InteropServices;
|
2014-09-09 18:46:13 +00:00
|
|
|
|
|
|
|
namespace FlatBuffers
|
|
|
|
{
|
|
|
|
/// <summary>
|
2016-08-13 01:01:38 +00:00
|
|
|
/// All tables in the generated code derive from this struct, and add their own accessors.
|
2014-09-09 18:46:13 +00:00
|
|
|
/// </summary>
|
2019-06-25 00:51:04 +00:00
|
|
|
public struct Table
|
2014-09-09 18:46:13 +00:00
|
|
|
{
|
2019-06-17 17:16:21 +00:00
|
|
|
public int bb_pos { get; private set; }
|
|
|
|
public ByteBuffer bb { get; private set; }
|
2014-09-09 18:46:13 +00:00
|
|
|
|
2015-11-10 05:20:16 +00:00
|
|
|
public ByteBuffer ByteBuffer { get { return bb; } }
|
|
|
|
|
2019-06-17 17:16:21 +00:00
|
|
|
// Re-init the internal state with an external buffer {@code ByteBuffer} and an offset within.
|
2019-12-24 01:35:54 +00:00
|
|
|
public Table(int _i, ByteBuffer _bb) : this()
|
2019-06-17 17:16:21 +00:00
|
|
|
{
|
|
|
|
bb = _bb;
|
|
|
|
bb_pos = _i;
|
|
|
|
}
|
|
|
|
|
2014-09-09 18:46:13 +00:00
|
|
|
// Look up a field in the vtable, return an offset into the object, or 0 if the field is not
|
|
|
|
// present.
|
2016-08-13 01:01:38 +00:00
|
|
|
public int __offset(int vtableOffset)
|
2016-08-14 11:58:51 +00:00
|
|
|
{
|
|
|
|
int vtable = bb_pos - bb.GetInt(bb_pos);
|
|
|
|
return vtableOffset < bb.GetShort(vtable) ? (int)bb.GetShort(vtable + vtableOffset) : 0;
|
|
|
|
}
|
2016-08-03 10:29:50 +00:00
|
|
|
|
2016-08-13 01:01:38 +00:00
|
|
|
public static int __offset(int vtableOffset, int offset, ByteBuffer bb)
|
2014-09-09 18:46:13 +00:00
|
|
|
{
|
2016-08-14 11:58:51 +00:00
|
|
|
int vtable = bb.Length - offset;
|
|
|
|
return (int)bb.GetShort(vtable + vtableOffset - bb.GetInt(vtable)) + vtable;
|
2014-09-09 18:46:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Retrieve the relative offset stored at "offset"
|
2016-08-13 01:01:38 +00:00
|
|
|
public int __indirect(int offset)
|
2014-09-09 18:46:13 +00:00
|
|
|
{
|
|
|
|
return offset + bb.GetInt(offset);
|
|
|
|
}
|
2016-08-13 01:01:38 +00:00
|
|
|
|
|
|
|
public static int __indirect(int offset, ByteBuffer bb)
|
2016-08-22 15:10:52 +00:00
|
|
|
{
|
|
|
|
return offset + bb.GetInt(offset);
|
|
|
|
}
|
2014-09-09 18:46:13 +00:00
|
|
|
|
|
|
|
// Create a .NET String from UTF-8 data stored inside the flatbuffer.
|
2016-08-13 01:01:38 +00:00
|
|
|
public string __string(int offset)
|
2014-09-09 18:46:13 +00:00
|
|
|
{
|
2016-08-14 11:58:51 +00:00
|
|
|
offset += bb.GetInt(offset);
|
|
|
|
var len = bb.GetInt(offset);
|
2014-09-09 18:46:13 +00:00
|
|
|
var startPos = offset + sizeof(int);
|
2018-03-12 18:30:46 +00:00
|
|
|
return bb.GetStringUTF8(startPos, len);
|
2014-09-09 18:46:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get the length of a vector whose offset is stored at "offset" in this object.
|
2016-08-13 01:01:38 +00:00
|
|
|
public int __vector_len(int offset)
|
2014-09-09 18:46:13 +00:00
|
|
|
{
|
|
|
|
offset += bb_pos;
|
|
|
|
offset += bb.GetInt(offset);
|
|
|
|
return bb.GetInt(offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the start of data of a vector whose offset is stored at "offset" in this object.
|
2016-08-13 01:01:38 +00:00
|
|
|
public int __vector(int offset)
|
2014-09-09 18:46:13 +00:00
|
|
|
{
|
|
|
|
offset += bb_pos;
|
|
|
|
return offset + bb.GetInt(offset) + sizeof(int); // data starts after the length
|
|
|
|
}
|
|
|
|
|
2020-09-25 17:39:28 +00:00
|
|
|
#if ENABLE_SPAN_T && (UNSAFE_BYTEBUFFER || NETSTANDARD2_1)
|
2018-08-23 17:05:31 +00:00
|
|
|
// Get the data of a vector whoses offset is stored at "offset" in this object as an
|
|
|
|
// Spant<byte>. If the vector is not present in the ByteBuffer,
|
|
|
|
// then an empty span will be returned.
|
2019-09-11 00:07:46 +00:00
|
|
|
public Span<T> __vector_as_span<T>(int offset, int elementSize) where T : struct
|
2018-08-23 17:05:31 +00:00
|
|
|
{
|
2019-09-11 00:07:46 +00:00
|
|
|
if (!BitConverter.IsLittleEndian)
|
|
|
|
{
|
|
|
|
throw new NotSupportedException("Getting typed span on a Big Endian " +
|
|
|
|
"system is not support");
|
|
|
|
}
|
|
|
|
|
2018-08-23 17:05:31 +00:00
|
|
|
var o = this.__offset(offset);
|
|
|
|
if (0 == o)
|
|
|
|
{
|
2019-09-11 00:07:46 +00:00
|
|
|
return new Span<T>();
|
2018-08-23 17:05:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var pos = this.__vector(o);
|
|
|
|
var len = this.__vector_len(o);
|
2019-09-11 00:07:46 +00:00
|
|
|
return MemoryMarshal.Cast<byte, T>(bb.ToSpan(pos, len * elementSize));
|
2018-08-23 17:05:31 +00:00
|
|
|
}
|
|
|
|
#else
|
2015-10-05 02:57:39 +00:00
|
|
|
// Get the data of a vector whoses offset is stored at "offset" in this object as an
|
|
|
|
// ArraySegment<byte>. If the vector is not present in the ByteBuffer,
|
|
|
|
// then a null value will be returned.
|
2016-08-13 01:01:38 +00:00
|
|
|
public ArraySegment<byte>? __vector_as_arraysegment(int offset)
|
|
|
|
{
|
2015-10-05 02:57:39 +00:00
|
|
|
var o = this.__offset(offset);
|
|
|
|
if (0 == o)
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
var pos = this.__vector(o);
|
|
|
|
var len = this.__vector_len(o);
|
2018-03-12 18:30:46 +00:00
|
|
|
return bb.ToArraySegment(pos, len);
|
2015-10-05 02:57:39 +00:00
|
|
|
}
|
2018-08-23 17:05:31 +00:00
|
|
|
#endif
|
2015-10-05 02:57:39 +00:00
|
|
|
|
2018-08-20 23:31:44 +00:00
|
|
|
// Get the data of a vector whoses offset is stored at "offset" in this object as an
|
|
|
|
// T[]. If the vector is not present in the ByteBuffer, then a null value will be
|
|
|
|
// returned.
|
|
|
|
public T[] __vector_as_array<T>(int offset)
|
|
|
|
where T : struct
|
|
|
|
{
|
|
|
|
if(!BitConverter.IsLittleEndian)
|
|
|
|
{
|
|
|
|
throw new NotSupportedException("Getting typed arrays on a Big Endian " +
|
|
|
|
"system is not support");
|
|
|
|
}
|
|
|
|
|
|
|
|
var o = this.__offset(offset);
|
|
|
|
if (0 == o)
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
var pos = this.__vector(o);
|
|
|
|
var len = this.__vector_len(o);
|
|
|
|
return bb.ToArray<T>(pos, len);
|
|
|
|
}
|
|
|
|
|
2014-09-09 18:46:13 +00:00
|
|
|
// Initialize any Table-derived type to point to the union at the given offset.
|
2016-08-13 01:01:38 +00:00
|
|
|
public T __union<T>(int offset) where T : struct, IFlatbufferObject
|
2014-09-09 18:46:13 +00:00
|
|
|
{
|
2016-08-13 01:01:38 +00:00
|
|
|
T t = new T();
|
2019-09-23 16:22:43 +00:00
|
|
|
t.__init(__indirect(offset), bb);
|
2014-09-09 18:46:13 +00:00
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2016-08-13 01:01:38 +00:00
|
|
|
public static bool __has_identifier(ByteBuffer bb, string ident)
|
2014-09-09 18:46:13 +00:00
|
|
|
{
|
|
|
|
if (ident.Length != FlatBufferConstants.FileIdentifierLength)
|
|
|
|
throw new ArgumentException("FlatBuffers: file identifier must be length " + FlatBufferConstants.FileIdentifierLength, "ident");
|
|
|
|
|
|
|
|
for (var i = 0; i < FlatBufferConstants.FileIdentifierLength; i++)
|
|
|
|
{
|
2015-05-06 14:33:50 +00:00
|
|
|
if (ident[i] != (char)bb.Get(bb.Position + sizeof(int) + i)) return false;
|
2014-09-09 18:46:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2016-08-13 01:01:38 +00:00
|
|
|
|
2016-08-14 11:58:51 +00:00
|
|
|
// Compare strings in the ByteBuffer.
|
2016-08-13 01:01:38 +00:00
|
|
|
public static int CompareStrings(int offset_1, int offset_2, ByteBuffer bb)
|
2016-08-14 11:58:51 +00:00
|
|
|
{
|
|
|
|
offset_1 += bb.GetInt(offset_1);
|
|
|
|
offset_2 += bb.GetInt(offset_2);
|
|
|
|
var len_1 = bb.GetInt(offset_1);
|
|
|
|
var len_2 = bb.GetInt(offset_2);
|
|
|
|
var startPos_1 = offset_1 + sizeof(int);
|
|
|
|
var startPos_2 = offset_2 + sizeof(int);
|
|
|
|
var len = Math.Min(len_1, len_2);
|
|
|
|
for(int i = 0; i < len; i++) {
|
2018-03-12 18:30:46 +00:00
|
|
|
byte b1 = bb.Get(i + startPos_1);
|
|
|
|
byte b2 = bb.Get(i + startPos_2);
|
|
|
|
if (b1 != b2)
|
|
|
|
return b1 - b2;
|
2016-08-14 11:58:51 +00:00
|
|
|
}
|
2016-08-26 16:41:32 +00:00
|
|
|
return len_1 - len_2;
|
2016-08-14 11:58:51 +00:00
|
|
|
}
|
2016-08-13 01:01:38 +00:00
|
|
|
|
2016-08-22 15:10:52 +00:00
|
|
|
// Compare string from the ByteBuffer with the string object
|
2016-08-13 01:01:38 +00:00
|
|
|
public static int CompareStrings(int offset_1, byte[] key, ByteBuffer bb)
|
2016-08-22 15:10:52 +00:00
|
|
|
{
|
|
|
|
offset_1 += bb.GetInt(offset_1);
|
|
|
|
var len_1 = bb.GetInt(offset_1);
|
|
|
|
var len_2 = key.Length;
|
|
|
|
var startPos_1 = offset_1 + sizeof(int);
|
|
|
|
var len = Math.Min(len_1, len_2);
|
|
|
|
for (int i = 0; i < len; i++) {
|
2018-03-12 18:30:46 +00:00
|
|
|
byte b = bb.Get(i + startPos_1);
|
|
|
|
if (b != key[i])
|
|
|
|
return b - key[i];
|
2016-08-22 15:10:52 +00:00
|
|
|
}
|
2016-08-26 16:41:32 +00:00
|
|
|
return len_1 - len_2;
|
2016-08-22 15:10:52 +00:00
|
|
|
}
|
2014-09-09 18:46:13 +00:00
|
|
|
}
|
|
|
|
}
|