blob: 02a19ea8067fc5654259743c0360c82dc7ff0fcf [file] [log] [blame]
(*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.
*)
{$SCOPEDENUMS ON}
unit Thrift.Protocol.Compact;
interface
uses
Classes,
SysUtils,
Math,
Generics.Collections,
Thrift.Configuration,
Thrift.Transport,
Thrift.Protocol,
Thrift.Utils;
type
ICompactProtocol = interface( IProtocol)
['{C01927EC-021A-45F7-93B1-23D6A5420EDD}']
end;
// Compact protocol implementation for thrift.
// Adapted from the C# version.
TCompactProtocolImpl = class( TProtocolImpl, ICompactProtocol)
public
type
TFactory = class( TInterfacedObject, IProtocolFactory)
public
function GetProtocol( const trans: ITransport): IProtocol;
end;
strict private const
{ TODO
static TStruct ANONYMOUS_STRUCT = new TStruct("");
static TField TSTOP = new TField("", TType.Stop, (short)0);
}
PROTOCOL_ID = Byte( $82);
VERSION = Byte( 1);
VERSION_MASK = Byte( $1F); // 0001 1111
TYPE_MASK = Byte( $E0); // 1110 0000
TYPE_BITS = Byte( $07); // 0000 0111
TYPE_SHIFT_AMOUNT = Byte( 5);
strict private type
// All of the on-wire type codes.
Types = (
STOP = $00,
BOOLEAN_TRUE = $01,
BOOLEAN_FALSE = $02,
BYTE_ = $03,
I16 = $04,
I32 = $05,
I64 = $06,
DOUBLE_ = $07,
BINARY = $08,
LIST = $09,
SET_ = $0A,
MAP = $0B,
STRUCT = $0C,
UUID = $0D
);
private type
TEightBytesArray = packed array[0..7] of Byte;
strict private const
ttypeToCompactType : array[TType] of Types = (
Types.STOP, // Stop = 0,
Types(-1), // Void = 1,
Types.BOOLEAN_TRUE, // Bool_ = 2,
Types.BYTE_, // Byte_ = 3,
Types.DOUBLE_, // Double_ = 4,
Types(-5), // unused
Types.I16, // I16 = 6,
Types(-7), // unused
Types.I32, // I32 = 8,
Types(-9), // unused
Types.I64, // I64 = 10,
Types.BINARY, // String_ = 11,
Types.STRUCT, // Struct = 12,
Types.MAP, // Map = 13,
Types.SET_, // Set_ = 14,
Types.LIST, // List = 15,
Types.UUID // Uuid = 16
);
tcompactTypeToType : array[Types] of TType = (
TType.Stop, // STOP
TType.Bool_, // BOOLEAN_TRUE
TType.Bool_, // BOOLEAN_FALSE
TType.Byte_, // BYTE_
TType.I16, // I16
TType.I32, // I32
TType.I64, // I64
TType.Double_, // DOUBLE_
TType.String_, // BINARY
TType.List, // LIST
TType.Set_, // SET_
TType.Map, // MAP
TType.Struct, // STRUCT
TType.Uuid // UUID
);
strict private
// Used to keep track of the last field for the current and previous structs,
// so we can do the delta stuff.
lastField_ : TStack<Integer>;
lastFieldId_ : Integer;
// If we encounter a boolean field begin, save the TField here so it can
// have the value incorporated.
strict private booleanField_ : TThriftField;
// If we Read a field header, and it's a boolean field, save the boolean
// value here so that ReadBool can use it.
strict private boolValue_ : ( unused, bool_true, bool_false);
public
constructor Create(const trans : ITransport); override;
destructor Destroy; override;
strict private
procedure WriteByteDirect( const b : Byte); overload;
// Writes a byte without any possibility of all that field header nonsense.
procedure WriteByteDirect( const n : Integer); overload;
// Write an i32 as a varint. Results in 1-5 bytes on the wire.
// TODO: make a permanent buffer like WriteVarint64?
procedure WriteVarint32( n : Cardinal);
strict private
// The workhorse of WriteFieldBegin. It has the option of doing a 'type override'
// of the type header. This is used specifically in the boolean field case.
procedure WriteFieldBeginInternal( const field : TThriftField; typeOverride : Byte);
public
procedure WriteMessageBegin( const msg: TThriftMessage); override;
procedure WriteMessageEnd; override;
procedure WriteStructBegin( const struc: TThriftStruct); override;
procedure WriteStructEnd; override;
procedure WriteFieldBegin( const field: TThriftField); override;
procedure WriteFieldEnd; override;
procedure WriteFieldStop; override;
procedure WriteMapBegin( const map: TThriftMap); override;
procedure WriteMapEnd; override;
procedure WriteListBegin( const list: TThriftList); override;
procedure WriteListEnd(); override;
procedure WriteSetBegin( const set_: TThriftSet ); override;
procedure WriteSetEnd(); override;
procedure WriteBool( b: Boolean); override;
procedure WriteByte( b: ShortInt); override;
procedure WriteI16( i16: SmallInt); override;
procedure WriteI32( i32: Integer); override;
procedure WriteI64( const i64: Int64); override;
procedure WriteDouble( const dub: Double); override;
procedure WriteBinary( const b: TBytes); overload; override;
procedure WriteUuid( const uuid: TGuid); override;
private // unit visible stuff
class function DoubleToInt64Bits( const db : Double) : Int64;
class function Int64BitsToDouble( const i64 : Int64) : Double;
// Abstract method for writing the start of lists and sets. List and sets on
// the wire differ only by the type indicator.
procedure WriteCollectionBegin( const elemType : TType; size : Integer);
procedure WriteVarint64( n : UInt64);
// Convert l into a zigzag long. This allows negative numbers to be
// represented compactly as a varint.
class function longToZigzag( const n : Int64) : UInt64;
// Convert n into a zigzag int. This allows negative numbers to be
// represented compactly as a varint.
class function intToZigZag( const n : Integer) : Cardinal;
//Convert a Int64 into little-endian bytes in buf starting at off and going until off+7.
class procedure fixedLongToBytes( const n : Int64; var buf : TEightBytesArray); inline;
strict protected
function GetMinSerializedSize( const aType : TType) : Integer; override;
procedure Reset; override;
public
function ReadMessageBegin: TThriftMessage; override;
procedure ReadMessageEnd(); override;
function ReadStructBegin: TThriftStruct; override;
procedure ReadStructEnd; override;
function ReadFieldBegin: TThriftField; override;
procedure ReadFieldEnd(); override;
function ReadMapBegin: TThriftMap; override;
procedure ReadMapEnd(); override;
function ReadListBegin: TThriftList; override;
procedure ReadListEnd(); override;
function ReadSetBegin: TThriftSet; override;
procedure ReadSetEnd(); override;
function ReadBool: Boolean; override;
function ReadByte: ShortInt; override;
function ReadI16: SmallInt; override;
function ReadI32: Integer; override;
function ReadI64: Int64; override;
function ReadDouble:Double; override;
function ReadBinary: TBytes; overload; override;
function ReadUuid: TGuid; override;
private
// Internal Reading methods
// Read an i32 from the wire as a varint. The MSB of each byte is set
// if there is another byte to follow. This can Read up to 5 bytes.
function ReadVarint32 : Cardinal;
// Read an i64 from the wire as a proper varint. The MSB of each byte is set
// if there is another byte to follow. This can Read up to 10 bytes.
function ReadVarint64 : UInt64;
// encoding helpers
// Convert from zigzag Integer to Integer.
class function zigzagToInt( const n : Cardinal ) : Integer;
// Convert from zigzag Int64 to Int64.
class function zigzagToLong( const n : UInt64) : Int64;
// Note that it's important that the mask bytes are Int64 literals,
// otherwise they'll default to ints, and when you shift an Integer left 56 bits,
// you just get a messed up Integer.
class function bytesToLong( const bytes : TEightBytesArray) : Int64; inline;
// type testing and converting
class function isBoolType( const b : byte) : Boolean;
// Given a TCompactProtocol.Types constant, convert it to its corresponding TType value.
class function getTType( const type_ : byte) : TType;
// Given a TType value, find the appropriate TCompactProtocol.Types constant.
class function getCompactType( const ttype : TType) : Byte;
end;
implementation
//--- TCompactProtocolImpl.TFactory ----------------------------------------
function TCompactProtocolImpl.TFactory.GetProtocol( const trans: ITransport): IProtocol;
begin
result := TCompactProtocolImpl.Create( trans);
end;
//--- TCompactProtocolImpl -------------------------------------------------
constructor TCompactProtocolImpl.Create( const trans : ITransport);
begin
inherited Create( trans);
lastFieldId_ := 0;
lastField_ := TStack<Integer>.Create;
Init( booleanField_, '', TType.Stop, 0);
boolValue_ := unused;
end;
destructor TCompactProtocolImpl.Destroy;
begin
try
FreeAndNil( lastField_);
finally
inherited Destroy;
end;
end;
procedure TCompactProtocolImpl.Reset;
begin
inherited Reset;
lastField_.Clear();
lastFieldId_ := 0;
Init( booleanField_, '', TType.Stop, 0);
boolValue_ := unused;
end;
// Writes a byte without any possibility of all that field header nonsense.
// Used internally by other writing methods that know they need to Write a byte.
procedure TCompactProtocolImpl.WriteByteDirect( const b : Byte);
begin
Transport.Write( @b, SizeOf(b));
end;
// Writes a byte without any possibility of all that field header nonsense.
procedure TCompactProtocolImpl.WriteByteDirect( const n : Integer);
begin
WriteByteDirect( Byte(n));
end;
// Write an i32 as a varint. Results in 1-5 bytes on the wire.
procedure TCompactProtocolImpl.WriteVarint32( n : Cardinal);
var idx : Integer;
i32buf : packed array[0..4] of Byte;
begin
idx := 0;
while TRUE do begin
ASSERT( idx < Length(i32buf));
// last part?
if ((n and not $7F) = 0) then begin
i32buf[idx] := Byte(n);
Inc(idx);
Break;
end;
i32buf[idx] := Byte((n and $7F) or $80);
Inc(idx);
n := n shr 7;
end;
Transport.Write( @i32buf[0], 0, idx);
end;
// Write a message header to the wire. Compact Protocol messages contain the
// protocol version so we can migrate forwards in the future if need be.
procedure TCompactProtocolImpl.WriteMessageBegin( const msg: TThriftMessage);
var versionAndType : Byte;
begin
Reset;
versionAndType := Byte( VERSION and VERSION_MASK)
or Byte( (Cardinal(msg.Type_) shl TYPE_SHIFT_AMOUNT) and TYPE_MASK);
WriteByteDirect( PROTOCOL_ID);
WriteByteDirect( versionAndType);
WriteVarint32( Cardinal(msg.SeqID));
WriteString( msg.Name);
end;
// Write a struct begin. This doesn't actually put anything on the wire. We use it as an
// opportunity to put special placeholder markers on the field stack so we can get the
// field id deltas correct.
procedure TCompactProtocolImpl.WriteStructBegin( const struc: TThriftStruct);
begin
lastField_.Push(lastFieldId_);
lastFieldId_ := 0;
end;
// Write a struct end. This doesn't actually put anything on the wire. We use this as an
// opportunity to pop the last field from the current struct off of the field stack.
procedure TCompactProtocolImpl.WriteStructEnd;
begin
lastFieldId_ := lastField_.Pop();
end;
// Write a field header containing the field id and field type. If the difference between the
// current field id and the last one is small (< 15), then the field id will be encoded in
// the 4 MSB as a delta. Otherwise, the field id will follow the type header as a zigzag varint.
procedure TCompactProtocolImpl.WriteFieldBegin( const field: TThriftField);
begin
case field.Type_ of
TType.Bool_ : booleanField_ := field; // we want to possibly include the value, so we'll wait.
else
WriteFieldBeginInternal(field, $FF);
end;
end;
// The workhorse of WriteFieldBegin. It has the option of doing a 'type override'
// of the type header. This is used specifically in the boolean field case.
procedure TCompactProtocolImpl.WriteFieldBeginInternal( const field : TThriftField; typeOverride : Byte);
var typeToWrite : Byte;
begin
// if there's a type override, use that.
if typeOverride = $FF
then typeToWrite := getCompactType( field.Type_)
else typeToWrite := typeOverride;
// check if we can use delta encoding for the field id
if (field.ID > lastFieldId_) and ((field.ID - lastFieldId_) <= 15)
then begin
// Write them together
WriteByteDirect( ((field.ID - lastFieldId_) shl 4) or typeToWrite);
end
else begin
// Write them separate
WriteByteDirect( typeToWrite);
WriteI16( field.ID);
end;
lastFieldId_ := field.ID;
end;
// Write the STOP symbol so we know there are no more fields in this struct.
procedure TCompactProtocolImpl.WriteFieldStop;
begin
WriteByteDirect( Byte( Types.STOP));
end;
// Write a map header. If the map is empty, omit the key and value type
// headers, as we don't need any additional information to skip it.
procedure TCompactProtocolImpl.WriteMapBegin( const map: TThriftMap);
var key, val : Byte;
begin
if (map.Count = 0)
then WriteByteDirect( 0)
else begin
WriteVarint32( Cardinal( map.Count));
key := getCompactType(map.KeyType);
val := getCompactType(map.ValueType);
WriteByteDirect( (key shl 4) or val);
end;
end;
// Write a list header.
procedure TCompactProtocolImpl.WriteListBegin( const list: TThriftList);
begin
WriteCollectionBegin( list.ElementType, list.Count);
end;
// Write a set header.
procedure TCompactProtocolImpl.WriteSetBegin( const set_: TThriftSet );
begin
WriteCollectionBegin( set_.ElementType, set_.Count);
end;
// Write a boolean value. Potentially, this could be a boolean field, in
// which case the field header info isn't written yet. If so, decide what the
// right type header is for the value and then Write the field header.
// Otherwise, Write a single byte.
procedure TCompactProtocolImpl.WriteBool( b: Boolean);
var bt : Types;
begin
if b
then bt := Types.BOOLEAN_TRUE
else bt := Types.BOOLEAN_FALSE;
if booleanField_.Type_ = TType.Bool_ then begin
// we haven't written the field header yet
WriteFieldBeginInternal( booleanField_, Byte(bt));
booleanField_.Type_ := TType.Stop;
end
else begin
// we're not part of a field, so just Write the value.
WriteByteDirect( Byte(bt));
end;
end;
// Write a byte. Nothing to see here!
procedure TCompactProtocolImpl.WriteByte( b: ShortInt);
begin
WriteByteDirect( Byte(b));
end;
// Write an I16 as a zigzag varint.
procedure TCompactProtocolImpl.WriteI16( i16: SmallInt);
begin
WriteVarint32( intToZigZag( i16));
end;
// Write an i32 as a zigzag varint.
procedure TCompactProtocolImpl.WriteI32( i32: Integer);
begin
WriteVarint32( intToZigZag( i32));
end;
// Write an i64 as a zigzag varint.
procedure TCompactProtocolImpl.WriteI64( const i64: Int64);
begin
WriteVarint64( longToZigzag( i64));
end;
class function TCompactProtocolImpl.DoubleToInt64Bits( const db : Double) : Int64;
begin
ASSERT( SizeOf(db) = SizeOf(result));
Move( db, result, SizeOf(result));
end;
class function TCompactProtocolImpl.Int64BitsToDouble( const i64 : Int64) : Double;
begin
ASSERT( SizeOf(i64) = SizeOf(result));
Move( i64, result, SizeOf(result));
end;
// Write a double to the wire as 8 bytes.
procedure TCompactProtocolImpl.WriteDouble( const dub: Double);
var data : TEightBytesArray;
begin
fixedLongToBytes( DoubleToInt64Bits(dub), data);
Transport.Write( @data[0], 0, SizeOf(data));
end;
// Write a byte array, using a varint for the size.
procedure TCompactProtocolImpl.WriteBinary( const b: TBytes);
begin
WriteVarint32( Cardinal(Length(b)));
Transport.Write( b);
end;
procedure TCompactProtocolImpl.WriteUuid( const uuid: TGuid);
var network : TGuid; // in network order (Big Endian)
begin
ASSERT( SizeOf(uuid) = 16);
network := uuid.SwapByteOrder;
Transport.Write( @network, 0, SizeOf(network));
end;
procedure TCompactProtocolImpl.WriteMessageEnd;
begin
// nothing to do
end;
procedure TCompactProtocolImpl.WriteMapEnd;
begin
// nothing to do
end;
procedure TCompactProtocolImpl.WriteListEnd;
begin
// nothing to do
end;
procedure TCompactProtocolImpl.WriteSetEnd;
begin
// nothing to do
end;
procedure TCompactProtocolImpl.WriteFieldEnd;
begin
// nothing to do
end;
// Abstract method for writing the start of lists and sets. List and sets on
// the wire differ only by the type indicator.
procedure TCompactProtocolImpl.WriteCollectionBegin( const elemType : TType; size : Integer);
begin
if size <= 14
then WriteByteDirect( (size shl 4) or getCompactType(elemType))
else begin
WriteByteDirect( $F0 or getCompactType(elemType));
WriteVarint32( Cardinal(size));
end;
end;
// Write an i64 as a varint. Results in 1-10 bytes on the wire.
procedure TCompactProtocolImpl.WriteVarint64( n : UInt64);
var idx : Integer;
varint64out : packed array[0..9] of Byte;
begin
idx := 0;
while TRUE do begin
ASSERT( idx < Length(varint64out));
// last one?
if (n and not UInt64($7F)) = 0 then begin
varint64out[idx] := Byte(n);
Inc(idx);
Break;
end;
varint64out[idx] := Byte((n and $7F) or $80);
Inc(idx);
n := n shr 7;
end;
Transport.Write( @varint64out[0], 0, idx);
end;
// Convert l into a zigzag Int64. This allows negative numbers to be
// represented compactly as a varint.
class function TCompactProtocolImpl.longToZigzag( const n : Int64) : UInt64;
begin
// there is no arithmetic right shift in Delphi
if n >= 0
then result := UInt64(n shl 1)
else result := UInt64(n shl 1) xor $FFFFFFFFFFFFFFFF;
end;
// Convert n into a zigzag Integer. This allows negative numbers to be
// represented compactly as a varint.
class function TCompactProtocolImpl.intToZigZag( const n : Integer) : Cardinal;
begin
// there is no arithmetic right shift in Delphi
if n >= 0
then result := Cardinal(n shl 1)
else result := Cardinal(n shl 1) xor $FFFFFFFF;
end;
// Convert a Int64 into 8 little-endian bytes in buf
class procedure TCompactProtocolImpl.fixedLongToBytes( const n : Int64; var buf : TEightBytesArray);
begin
ASSERT( Length(buf) >= 8);
buf[0] := Byte( n and $FF);
buf[1] := Byte((n shr 8) and $FF);
buf[2] := Byte((n shr 16) and $FF);
buf[3] := Byte((n shr 24) and $FF);
buf[4] := Byte((n shr 32) and $FF);
buf[5] := Byte((n shr 40) and $FF);
buf[6] := Byte((n shr 48) and $FF);
buf[7] := Byte((n shr 56) and $FF);
end;
// Read a message header.
function TCompactProtocolImpl.ReadMessageBegin : TThriftMessage;
var protocolId, versionAndType, version, type_ : Byte;
seqid : Integer;
msgNm : String;
begin
Reset;
protocolId := Byte( ReadByte);
if (protocolId <> PROTOCOL_ID)
then raise TProtocolExceptionBadVersion.Create( 'Expected protocol id ' + IntToHex(PROTOCOL_ID,2)
+ ' but got ' + IntToHex(protocolId,2));
versionAndType := Byte( ReadByte);
version := Byte( versionAndType and VERSION_MASK);
if (version <> VERSION)
then raise TProtocolExceptionBadVersion.Create( 'Expected version ' +IntToStr(VERSION)
+ ' but got ' + IntToStr(version));
type_ := Byte( (versionAndType shr TYPE_SHIFT_AMOUNT) and TYPE_BITS);
seqid := Integer( ReadVarint32);
msgNm := ReadString;
Init( result, msgNm, TMessageType(type_), seqid);
end;
// Read a struct begin. There's nothing on the wire for this, but it is our
// opportunity to push a new struct begin marker onto the field stack.
function TCompactProtocolImpl.ReadStructBegin: TThriftStruct;
begin
lastField_.Push( lastFieldId_);
lastFieldId_ := 0;
Init( result);
end;
// Doesn't actually consume any wire data, just removes the last field for
// this struct from the field stack.
procedure TCompactProtocolImpl.ReadStructEnd;
begin
// consume the last field we Read off the wire.
lastFieldId_ := lastField_.Pop();
end;
// Read a field header off the wire.
function TCompactProtocolImpl.ReadFieldBegin: TThriftField;
var type_ : Byte;
modifier : ShortInt;
fieldId : SmallInt;
begin
type_ := Byte( ReadByte);
// if it's a stop, then we can return immediately, as the struct is over.
if type_ = Byte(Types.STOP) then begin
Init( result, '', TType.Stop, 0);
Exit;
end;
// mask off the 4 MSB of the type header. it could contain a field id delta.
modifier := ShortInt( (type_ and $F0) shr 4);
if (modifier = 0)
then fieldId := ReadI16 // not a delta. look ahead for the zigzag varint field id.
else fieldId := SmallInt( lastFieldId_ + modifier); // add the delta to the last Read field id.
Init( result, '', getTType(Byte(type_ and $0F)), fieldId);
// if this happens to be a boolean field, the value is encoded in the type
// save the boolean value in a special instance variable.
if isBoolType(type_) then begin
if Byte(type_ and $0F) = Byte(Types.BOOLEAN_TRUE)
then boolValue_ := bool_true
else boolValue_ := bool_false;
end;
// push the new field onto the field stack so we can keep the deltas going.
lastFieldId_ := result.ID;
end;
// Read a map header off the wire. If the size is zero, skip Reading the key
// and value type. This means that 0-length maps will yield TMaps without the
// "correct" types.
function TCompactProtocolImpl.ReadMapBegin: TThriftMap;
var size : Integer;
keyAndValueType : Byte;
key, val : TType;
begin
size := Integer( ReadVarint32);
if size = 0
then keyAndValueType := 0
else keyAndValueType := Byte( ReadByte);
key := getTType( Byte( keyAndValueType shr 4));
val := getTType( Byte( keyAndValueType and $F));
Init( result, key, val, size);
ASSERT( (result.KeyType = key) and (result.ValueType = val));
CheckReadBytesAvailable(result);
end;
// Read a list header off the wire. If the list size is 0-14, the size will
// be packed into the element type header. If it's a longer list, the 4 MSB
// of the element type header will be $F, and a varint will follow with the
// true size.
function TCompactProtocolImpl.ReadListBegin: TThriftList;
var size_and_type : Byte;
size : Integer;
type_ : TType;
begin
size_and_type := Byte( ReadByte);
size := (size_and_type shr 4) and $0F;
if (size = 15)
then size := Integer( ReadVarint32);
type_ := getTType( size_and_type);
Init( result, type_, size);
CheckReadBytesAvailable(result);
end;
// Read a set header off the wire. If the set size is 0-14, the size will
// be packed into the element type header. If it's a longer set, the 4 MSB
// of the element type header will be $F, and a varint will follow with the
// true size.
function TCompactProtocolImpl.ReadSetBegin: TThriftSet;
var size_and_type : Byte;
size : Integer;
type_ : TType;
begin
size_and_type := Byte( ReadByte);
size := (size_and_type shr 4) and $0F;
if (size = 15)
then size := Integer( ReadVarint32);
type_ := getTType( size_and_type);
Init( result, type_, size);
CheckReadBytesAvailable(result);
end;
// Read a boolean off the wire. If this is a boolean field, the value should
// already have been Read during ReadFieldBegin, so we'll just consume the
// pre-stored value. Otherwise, Read a byte.
function TCompactProtocolImpl.ReadBool: Boolean;
begin
if boolValue_ <> unused then begin
result := (boolValue_ = bool_true);
boolValue_ := unused;
Exit;
end;
result := (Byte(ReadByte) = Byte(Types.BOOLEAN_TRUE));
end;
// Read a single byte off the wire. Nothing interesting here.
function TCompactProtocolImpl.ReadByte: ShortInt;
begin
Transport.ReadAll( @result, SizeOf(result), 0, 1);
end;
// Read an i16 from the wire as a zigzag varint.
function TCompactProtocolImpl.ReadI16: SmallInt;
begin
result := SmallInt( zigzagToInt( ReadVarint32));
end;
// Read an i32 from the wire as a zigzag varint.
function TCompactProtocolImpl.ReadI32: Integer;
begin
result := zigzagToInt( ReadVarint32);
end;
// Read an i64 from the wire as a zigzag varint.
function TCompactProtocolImpl.ReadI64: Int64;
begin
result := zigzagToLong( ReadVarint64);
end;
// No magic here - just Read a double off the wire.
function TCompactProtocolImpl.ReadDouble : Double;
var longBits : TEightBytesArray;
begin
ASSERT( SizeOf(longBits) = SizeOf(result));
Transport.ReadAll( @longBits[0], SizeOf(longBits), 0, SizeOf(longBits));
result := Int64BitsToDouble( bytesToLong( longBits));
end;
// Read a byte[] from the wire.
function TCompactProtocolImpl.ReadBinary: TBytes;
var length : Integer;
begin
length := Integer( ReadVarint32);
FTrans.CheckReadBytesAvailable(length);
SetLength( result, length);
if (length > 0)
then Transport.ReadAll( result, 0, length);
end;
function TCompactProtocolImpl.ReadUuid: TGuid;
var network : TGuid; // in network order (Big Endian)
begin
ASSERT( SizeOf(result) = 16);
FTrans.ReadAll( @network, SizeOf(network), 0, SizeOf(network));
result := network.SwapByteOrder;
end;
procedure TCompactProtocolImpl.ReadMessageEnd;
begin
// nothing to do
end;
procedure TCompactProtocolImpl.ReadFieldEnd;
begin
// nothing to do
end;
procedure TCompactProtocolImpl.ReadMapEnd;
begin
// nothing to do
end;
procedure TCompactProtocolImpl.ReadListEnd;
begin
// nothing to do
end;
procedure TCompactProtocolImpl.ReadSetEnd;
begin
// nothing to do
end;
// Read an i32 from the wire as a varint. The MSB of each byte is set
// if there is another byte to follow. This can Read up to 5 bytes.
function TCompactProtocolImpl.ReadVarint32 : Cardinal;
var shift : Integer;
b : Byte;
begin
result := 0;
shift := 0;
while TRUE do begin
b := Byte( ReadByte);
result := result or (Cardinal(b and $7F) shl shift);
if ((b and $80) <> $80)
then Break;
Inc( shift, 7);
end;
end;
// Read an i64 from the wire as a proper varint. The MSB of each byte is set
// if there is another byte to follow. This can Read up to 10 bytes.
function TCompactProtocolImpl.ReadVarint64 : UInt64;
var shift : Integer;
b : Byte;
begin
result := 0;
shift := 0;
while TRUE do begin
b := Byte( ReadByte);
result := result or (UInt64(b and $7F) shl shift);
if ((b and $80) <> $80)
then Break;
Inc( shift, 7);
end;
end;
// Convert from zigzag Integer to Integer.
class function TCompactProtocolImpl.zigzagToInt( const n : Cardinal ) : Integer;
begin
result := Integer(n shr 1) xor (-Integer(n and 1));
end;
// Convert from zigzag Int64 to Int64.
class function TCompactProtocolImpl.zigzagToLong( const n : UInt64) : Int64;
begin
result := Int64(n shr 1) xor (-Int64(n and 1));
end;
// Note that it's important that the mask bytes are Int64 literals,
// otherwise they'll default to ints, and when you shift an Integer left 56 bits,
// you just get a messed up Integer.
class function TCompactProtocolImpl.bytesToLong( const bytes : TEightBytesArray) : Int64;
begin
ASSERT( Length(bytes) >= 8);
result := (Int64(bytes[7] and $FF) shl 56) or
(Int64(bytes[6] and $FF) shl 48) or
(Int64(bytes[5] and $FF) shl 40) or
(Int64(bytes[4] and $FF) shl 32) or
(Int64(bytes[3] and $FF) shl 24) or
(Int64(bytes[2] and $FF) shl 16) or
(Int64(bytes[1] and $FF) shl 8) or
(Int64(bytes[0] and $FF));
end;
class function TCompactProtocolImpl.isBoolType( const b : byte) : Boolean;
var lowerNibble : Byte;
begin
lowerNibble := b and $0f;
result := (Types(lowerNibble) in [Types.BOOLEAN_TRUE, Types.BOOLEAN_FALSE]);
end;
// Given a TCompactProtocol.Types constant, convert it to its corresponding TType value.
class function TCompactProtocolImpl.getTType( const type_ : byte) : TType;
var tct : Types;
begin
tct := Types( type_ and $0F);
if tct in [Low(Types)..High(Types)]
then result := tcompactTypeToType[tct]
else raise TProtocolExceptionInvalidData.Create('don''t know what type: '+IntToStr(Ord(tct)));
end;
// Given a TType value, find the appropriate TCompactProtocol.Types constant.
class function TCompactProtocolImpl.getCompactType( const ttype : TType) : Byte;
begin
if ttype in VALID_TTYPES
then result := Byte( ttypeToCompactType[ttype])
else raise TProtocolExceptionInvalidData.Create('don''t know what type: '+IntToStr(Ord(ttype)));
end;
function TCompactProtocolImpl.GetMinSerializedSize( const aType : TType) : Integer;
// Return the minimum number of bytes a type will consume on the wire
begin
case aType of
TType.Stop: result := 0;
TType.Void: result := 0;
TType.Bool_: result := SizeOf(Byte);
TType.Byte_: result := SizeOf(Byte);
TType.Double_: result := 8; // uses fixedLongToBytes() which always writes 8 bytes
TType.I16: result := SizeOf(Byte);
TType.I32: result := SizeOf(Byte);
TType.I64: result := SizeOf(Byte);
TType.String_: result := SizeOf(Byte); // string length
TType.Struct: result := 0; // empty struct
TType.Map: result := SizeOf(Byte); // element count
TType.Set_: result := SizeOf(Byte); // element count
TType.List: result := SizeOf(Byte); // element count
TType.Uuid: result := SizeOf(TGuid);
else
raise TTransportExceptionBadArgs.Create('Unhandled type code');
end;
end;
//--- unit tests -------------------------------------------
{$IFDEF Debug}
procedure TestDoubleToInt64Bits;
procedure TestPair( const a : Double; const b : Int64);
begin
ASSERT( TCompactProtocolImpl.DoubleToInt64Bits(a) = b);
ASSERT( TCompactProtocolImpl.Int64BitsToDouble(b) = a);
end;
begin
TestPair( 1.0000000000000000E+000, Int64($3FF0000000000000));
TestPair( 1.5000000000000000E+001, Int64($402E000000000000));
TestPair( 2.5500000000000000E+002, Int64($406FE00000000000));
TestPair( 4.2949672950000000E+009, Int64($41EFFFFFFFE00000));
TestPair( 3.9062500000000000E-003, Int64($3F70000000000000));
TestPair( 2.3283064365386963E-010, Int64($3DF0000000000000));
TestPair( 1.2345678901230000E-300, Int64($01AA74FE1C1E7E45));
TestPair( 1.2345678901234500E-150, Int64($20D02A36586DB4BB));
TestPair( 1.2345678901234565E+000, Int64($3FF3C0CA428C59FA));
TestPair( 1.2345678901234567E+000, Int64($3FF3C0CA428C59FB));
TestPair( 1.2345678901234569E+000, Int64($3FF3C0CA428C59FC));
TestPair( 1.2345678901234569E+150, Int64($5F182344CD3CDF9F));
TestPair( 1.2345678901234569E+300, Int64($7E3D7EE8BCBBD352));
TestPair( -1.7976931348623157E+308, Int64($FFEFFFFFFFFFFFFF));
TestPair( 1.7976931348623157E+308, Int64($7FEFFFFFFFFFFFFF));
TestPair( 4.9406564584124654E-324, Int64($0000000000000001));
TestPair( 0.0000000000000000E+000, Int64($0000000000000000));
TestPair( 4.94065645841247E-324, Int64($0000000000000001));
TestPair( 3.2378592100206092E-319, Int64($000000000000FFFF));
TestPair( 1.3906711615669959E-309, Int64($0000FFFFFFFFFFFF));
TestPair( NegInfinity, Int64($FFF0000000000000));
TestPair( Infinity, Int64($7FF0000000000000));
// NaN is special
ASSERT( TCompactProtocolImpl.DoubleToInt64Bits( NaN) = Int64($FFF8000000000000));
ASSERT( IsNan( TCompactProtocolImpl.Int64BitsToDouble( Int64($FFF8000000000000))));
end;
{$ENDIF}
{$IFDEF Debug}
procedure TestZigZag;
procedure Test32( const test : Integer);
var zz : Cardinal;
begin
zz := TCompactProtocolImpl.intToZigZag(test);
ASSERT( TCompactProtocolImpl.zigzagToInt(zz) = test, IntToStr(test));
end;
procedure Test64( const test : Int64);
var zz : UInt64;
begin
zz := TCompactProtocolImpl.longToZigzag(test);
ASSERT( TCompactProtocolImpl.zigzagToLong(zz) = test, IntToStr(test));
end;
var i : Integer;
begin
// protobuf testcases
ASSERT( TCompactProtocolImpl.intToZigZag(0) = 0, 'pb #1 to ZigZag');
ASSERT( TCompactProtocolImpl.intToZigZag(-1) = 1, 'pb #2 to ZigZag');
ASSERT( TCompactProtocolImpl.intToZigZag(1) = 2, 'pb #3 to ZigZag');
ASSERT( TCompactProtocolImpl.intToZigZag(-2) = 3, 'pb #4 to ZigZag');
ASSERT( TCompactProtocolImpl.intToZigZag(+2147483647) = 4294967294, 'pb #5 to ZigZag');
ASSERT( TCompactProtocolImpl.intToZigZag(-2147483648) = 4294967295, 'pb #6 to ZigZag');
// protobuf testcases
ASSERT( TCompactProtocolImpl.zigzagToInt(0) = 0, 'pb #1 from ZigZag');
ASSERT( TCompactProtocolImpl.zigzagToInt(1) = -1, 'pb #2 from ZigZag');
ASSERT( TCompactProtocolImpl.zigzagToInt(2) = 1, 'pb #3 from ZigZag');
ASSERT( TCompactProtocolImpl.zigzagToInt(3) = -2, 'pb #4 from ZigZag');
ASSERT( TCompactProtocolImpl.zigzagToInt(4294967294) = +2147483647, 'pb #5 from ZigZag');
ASSERT( TCompactProtocolImpl.zigzagToInt(4294967295) = -2147483648, 'pb #6 from ZigZag');
// back and forth 32
Test32( 0);
for i := 0 to 30 do begin
Test32( +(Integer(1) shl i));
Test32( -(Integer(1) shl i));
end;
Test32( Integer($7FFFFFFF));
Test32( Integer($80000000));
// back and forth 64
Test64( 0);
for i := 0 to 62 do begin
Test64( +(Int64(1) shl i));
Test64( -(Int64(1) shl i));
end;
Test64( Int64($7FFFFFFFFFFFFFFF));
Test64( Int64($8000000000000000));
end;
{$ENDIF}
{$IFDEF Debug}
procedure TestLongBytes;
procedure Test( const test : Int64);
var buf : TCompactProtocolImpl.TEightBytesArray;
begin
TCompactProtocolImpl.fixedLongToBytes( test, buf);
ASSERT( TCompactProtocolImpl.bytesToLong( buf) = test, IntToStr(test));
end;
var i : Integer;
begin
Test( 0);
for i := 0 to 62 do begin
Test( +(Int64(1) shl i));
Test( -(Int64(1) shl i));
end;
Test( Int64($7FFFFFFFFFFFFFFF));
Test( Int64($8000000000000000));
end;
{$ENDIF}
{$IFDEF Debug}
procedure UnitTest;
var w : WORD;
const FPU_CW_DENORMALIZED = $0002;
begin
w := Get8087CW;
try
Set8087CW( w or FPU_CW_DENORMALIZED);
TestDoubleToInt64Bits;
TestZigZag;
TestLongBytes;
finally
Set8087CW( w);
end;
end;
{$ENDIF}
initialization
{$IFDEF Debug}
UnitTest;
{$ENDIF}
end.