blob: 9203865587d6986398a19fb915d16cbe99953164 [file] [log] [blame]
Mark Slee9f0c6512007-02-28 23:58:26 +00001// Copyright (c) 2006- Facebook
2// Distributed under the Thrift Software License
3//
4// See accompanying file LICENSE or visit the Thrift site at:
5// http://developers.facebook.com/thrift/
6
Mark Sleef5f2be42006-09-05 21:05:31 +00007#ifndef _THRIFT_PROTOCOL_TBINARYPROTOCOL_H_
8#define _THRIFT_PROTOCOL_TBINARYPROTOCOL_H_ 1
Mark Sleee8540632006-05-30 09:24:40 +00009
Marc Slemkod42a2c22006-08-10 03:30:18 +000010#include "TProtocol.h"
Marc Slemko16698852006-08-04 03:16:10 +000011
12#include <boost/shared_ptr.hpp>
Mark Sleee8540632006-05-30 09:24:40 +000013
David Reiss0c90f6f2008-02-06 22:18:40 +000014namespace facebook { namespace thrift { namespace protocol {
Marc Slemko6f038a72006-08-03 18:58:09 +000015
Mark Sleee8540632006-05-30 09:24:40 +000016/**
17 * The default binary protocol for thrift. Writes all data in a very basic
18 * binary format, essentially just spitting out the raw bytes.
19 *
20 * @author Mark Slee <mcslee@facebook.com>
21 */
Mark Sleef9831082007-02-20 20:59:21 +000022class TBinaryProtocol : public TProtocol {
Mark Slee808454e2007-06-20 21:51:57 +000023 protected:
24 static const int32_t VERSION_MASK = 0xffff0000;
25 static const int32_t VERSION_1 = 0x80010000;
David Reiss4e7530d2007-09-04 21:49:53 +000026 // VERSION_2 (0x80020000) is taken by TDenseProtocol.
Mark Slee808454e2007-06-20 21:51:57 +000027
Marc Slemko0b4ffa92006-08-11 02:49:29 +000028 public:
Mark Slee5ea15f92007-03-05 22:55:59 +000029 TBinaryProtocol(boost::shared_ptr<TTransport> trans) :
Mark Sleef9831082007-02-20 20:59:21 +000030 TProtocol(trans),
31 string_limit_(0),
32 container_limit_(0),
Mark Slee808454e2007-06-20 21:51:57 +000033 strict_read_(false),
34 strict_write_(true),
Mark Sleef9831082007-02-20 20:59:21 +000035 string_buf_(NULL),
36 string_buf_size_(0) {}
Mark Slee4af6ed72006-10-25 19:02:49 +000037
Mark Slee5ea15f92007-03-05 22:55:59 +000038 TBinaryProtocol(boost::shared_ptr<TTransport> trans,
Mark Sleef9831082007-02-20 20:59:21 +000039 int32_t string_limit,
Mark Slee808454e2007-06-20 21:51:57 +000040 int32_t container_limit,
41 bool strict_read,
42 bool strict_write) :
Mark Sleef9831082007-02-20 20:59:21 +000043 TProtocol(trans),
44 string_limit_(string_limit),
45 container_limit_(container_limit),
Mark Slee808454e2007-06-20 21:51:57 +000046 strict_read_(strict_read),
47 strict_write_(strict_write),
Mark Sleef9831082007-02-20 20:59:21 +000048 string_buf_(NULL),
49 string_buf_size_(0) {}
50
51 ~TBinaryProtocol() {
52 if (string_buf_ != NULL) {
53 free(string_buf_);
54 string_buf_size_ = 0;
55 }
56 }
57
58 void setStringSizeLimit(int32_t string_limit) {
59 string_limit_ = string_limit;
60 }
61
62 void setContainerSizeLimit(int32_t container_limit) {
63 container_limit_ = container_limit;
64 }
Mark Sleee8540632006-05-30 09:24:40 +000065
Mark Slee808454e2007-06-20 21:51:57 +000066 void setStrict(bool strict_read, bool strict_write) {
67 strict_read_ = strict_read;
68 strict_write_ = strict_write;
69 }
70
Mark Slee8d7e1f62006-06-07 06:48:56 +000071 /**
72 * Writing functions.
73 */
Mark Sleee8540632006-05-30 09:24:40 +000074
Mark Slee82a6c0f2007-04-04 21:08:21 +000075 virtual uint32_t writeMessageBegin(const std::string& name,
76 const TMessageType messageType,
77 const int32_t seqid);
Marc Slemko16698852006-08-04 03:16:10 +000078
Mark Slee4af6ed72006-10-25 19:02:49 +000079 virtual uint32_t writeMessageEnd();
Marc Slemko16698852006-08-04 03:16:10 +000080
81
Mark Slee4af6ed72006-10-25 19:02:49 +000082 uint32_t writeStructBegin(const std::string& name);
Mark Sleee8540632006-05-30 09:24:40 +000083
Mark Slee4af6ed72006-10-25 19:02:49 +000084 uint32_t writeStructEnd();
Mark Sleee8540632006-05-30 09:24:40 +000085
Mark Slee4af6ed72006-10-25 19:02:49 +000086 uint32_t writeFieldBegin(const std::string& name,
87 const TType fieldType,
88 const int16_t fieldId);
Mark Slee8d7e1f62006-06-07 06:48:56 +000089
Mark Slee4af6ed72006-10-25 19:02:49 +000090 uint32_t writeFieldEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +000091
Mark Slee4af6ed72006-10-25 19:02:49 +000092 uint32_t writeFieldStop();
David Reiss0c90f6f2008-02-06 22:18:40 +000093
Mark Slee4af6ed72006-10-25 19:02:49 +000094 uint32_t writeMapBegin(const TType keyType,
95 const TType valType,
96 const uint32_t size);
Mark Slee8d7e1f62006-06-07 06:48:56 +000097
Mark Slee4af6ed72006-10-25 19:02:49 +000098 uint32_t writeMapEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +000099
Mark Slee4af6ed72006-10-25 19:02:49 +0000100 uint32_t writeListBegin(const TType elemType,
101 const uint32_t size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000102
Mark Slee4af6ed72006-10-25 19:02:49 +0000103 uint32_t writeListEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000104
Mark Slee4af6ed72006-10-25 19:02:49 +0000105 uint32_t writeSetBegin(const TType elemType,
106 const uint32_t size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000107
Mark Slee4af6ed72006-10-25 19:02:49 +0000108 uint32_t writeSetEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000109
Mark Slee4af6ed72006-10-25 19:02:49 +0000110 uint32_t writeBool(const bool value);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000111
Mark Slee4af6ed72006-10-25 19:02:49 +0000112 uint32_t writeByte(const int8_t byte);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000113
Mark Slee4af6ed72006-10-25 19:02:49 +0000114 uint32_t writeI16(const int16_t i16);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000115
Mark Slee4af6ed72006-10-25 19:02:49 +0000116 uint32_t writeI32(const int32_t i32);
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000117
Mark Slee4af6ed72006-10-25 19:02:49 +0000118 uint32_t writeI64(const int64_t i64);
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000119
Mark Slee4af6ed72006-10-25 19:02:49 +0000120 uint32_t writeDouble(const double dub);
Mark Sleec98d0502006-09-06 02:42:25 +0000121
Mark Slee4af6ed72006-10-25 19:02:49 +0000122 uint32_t writeString(const std::string& str);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000123
David Reissc005b1b2008-02-15 01:38:18 +0000124 uint32_t writeBinary(const std::string& str);
125
Mark Slee8d7e1f62006-06-07 06:48:56 +0000126 /**
127 * Reading functions
128 */
129
Marc Slemko16698852006-08-04 03:16:10 +0000130
Mark Slee4af6ed72006-10-25 19:02:49 +0000131 uint32_t readMessageBegin(std::string& name,
David Reiss96d23882007-07-26 21:10:32 +0000132 TMessageType& messageType,
133 int32_t& seqid);
Marc Slemko16698852006-08-04 03:16:10 +0000134
Mark Slee4af6ed72006-10-25 19:02:49 +0000135 uint32_t readMessageEnd();
Marc Slemko16698852006-08-04 03:16:10 +0000136
Mark Slee4af6ed72006-10-25 19:02:49 +0000137 uint32_t readStructBegin(std::string& name);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000138
Mark Slee4af6ed72006-10-25 19:02:49 +0000139 uint32_t readStructEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000140
Mark Slee4af6ed72006-10-25 19:02:49 +0000141 uint32_t readFieldBegin(std::string& name,
David Reiss96d23882007-07-26 21:10:32 +0000142 TType& fieldType,
143 int16_t& fieldId);
David Reiss0c90f6f2008-02-06 22:18:40 +0000144
Mark Slee4af6ed72006-10-25 19:02:49 +0000145 uint32_t readFieldEnd();
David Reiss0c90f6f2008-02-06 22:18:40 +0000146
Mark Slee4af6ed72006-10-25 19:02:49 +0000147 uint32_t readMapBegin(TType& keyType,
David Reiss96d23882007-07-26 21:10:32 +0000148 TType& valType,
149 uint32_t& size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000150
Mark Slee4af6ed72006-10-25 19:02:49 +0000151 uint32_t readMapEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000152
Mark Slee4af6ed72006-10-25 19:02:49 +0000153 uint32_t readListBegin(TType& elemType,
154 uint32_t& size);
David Reiss0c90f6f2008-02-06 22:18:40 +0000155
Mark Slee4af6ed72006-10-25 19:02:49 +0000156 uint32_t readListEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000157
Mark Slee4af6ed72006-10-25 19:02:49 +0000158 uint32_t readSetBegin(TType& elemType,
David Reiss96d23882007-07-26 21:10:32 +0000159 uint32_t& size);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000160
Mark Slee4af6ed72006-10-25 19:02:49 +0000161 uint32_t readSetEnd();
Mark Slee8d7e1f62006-06-07 06:48:56 +0000162
Mark Slee4af6ed72006-10-25 19:02:49 +0000163 uint32_t readBool(bool& value);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000164
Mark Slee4af6ed72006-10-25 19:02:49 +0000165 uint32_t readByte(int8_t& byte);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000166
Mark Slee4af6ed72006-10-25 19:02:49 +0000167 uint32_t readI16(int16_t& i16);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000168
Mark Slee4af6ed72006-10-25 19:02:49 +0000169 uint32_t readI32(int32_t& i32);
Mark Slee8d7e1f62006-06-07 06:48:56 +0000170
Mark Slee4af6ed72006-10-25 19:02:49 +0000171 uint32_t readI64(int64_t& i64);
Marc Slemko0b4ffa92006-08-11 02:49:29 +0000172
Mark Slee4af6ed72006-10-25 19:02:49 +0000173 uint32_t readDouble(double& dub);
Mark Sleec98d0502006-09-06 02:42:25 +0000174
Mark Slee4af6ed72006-10-25 19:02:49 +0000175 uint32_t readString(std::string& str);
Mark Sleef9831082007-02-20 20:59:21 +0000176
David Reissc005b1b2008-02-15 01:38:18 +0000177 uint32_t readBinary(std::string& str);
178
Mark Slee808454e2007-06-20 21:51:57 +0000179 protected:
180 uint32_t readStringBody(std::string& str, int32_t sz);
181
Mark Sleef9831082007-02-20 20:59:21 +0000182 int32_t string_limit_;
183 int32_t container_limit_;
184
Mark Slee808454e2007-06-20 21:51:57 +0000185 // Enforce presence of version identifier
186 bool strict_read_;
187 bool strict_write_;
188
Mark Sleef9831082007-02-20 20:59:21 +0000189 // Buffer for reading strings, save for the lifetime of the protocol to
190 // avoid memory churn allocating memory on every string read
191 uint8_t* string_buf_;
192 int32_t string_buf_size_;
193
Mark Slee4af6ed72006-10-25 19:02:49 +0000194};
195
196/**
197 * Constructs binary protocol handlers
198 */
199class TBinaryProtocolFactory : public TProtocolFactory {
200 public:
Mark Sleef9831082007-02-20 20:59:21 +0000201 TBinaryProtocolFactory() :
202 string_limit_(0),
Mark Slee808454e2007-06-20 21:51:57 +0000203 container_limit_(0),
204 strict_read_(false),
205 strict_write_(true) {}
Mark Sleef9831082007-02-20 20:59:21 +0000206
Mark Slee808454e2007-06-20 21:51:57 +0000207 TBinaryProtocolFactory(int32_t string_limit, int32_t container_limit, bool strict_read, bool strict_write) :
Mark Sleef9831082007-02-20 20:59:21 +0000208 string_limit_(string_limit),
Mark Slee808454e2007-06-20 21:51:57 +0000209 container_limit_(container_limit),
210 strict_read_(strict_read),
211 strict_write_(strict_write) {}
Mark Slee4af6ed72006-10-25 19:02:49 +0000212
213 virtual ~TBinaryProtocolFactory() {}
214
Mark Sleef9831082007-02-20 20:59:21 +0000215 void setStringSizeLimit(int32_t string_limit) {
216 string_limit_ = string_limit;
Mark Slee4af6ed72006-10-25 19:02:49 +0000217 }
Mark Sleef9831082007-02-20 20:59:21 +0000218
219 void setContainerSizeLimit(int32_t container_limit) {
220 container_limit_ = container_limit;
221 }
222
Mark Slee808454e2007-06-20 21:51:57 +0000223 void setStrict(bool strict_read, bool strict_write) {
224 strict_read_ = strict_read;
225 strict_write_ = strict_write;
226 }
227
Mark Sleef9831082007-02-20 20:59:21 +0000228 boost::shared_ptr<TProtocol> getProtocol(boost::shared_ptr<TTransport> trans) {
Mark Slee808454e2007-06-20 21:51:57 +0000229 return boost::shared_ptr<TProtocol>(new TBinaryProtocol(trans, string_limit_, container_limit_, strict_read_, strict_write_));
Mark Sleef9831082007-02-20 20:59:21 +0000230 }
231
232 private:
233 int32_t string_limit_;
234 int32_t container_limit_;
Mark Slee808454e2007-06-20 21:51:57 +0000235 bool strict_read_;
236 bool strict_write_;
Mark Sleef9831082007-02-20 20:59:21 +0000237
Mark Sleee8540632006-05-30 09:24:40 +0000238};
239
Marc Slemko6f038a72006-08-03 18:58:09 +0000240}}} // facebook::thrift::protocol
241
Mark Sleef5f2be42006-09-05 21:05:31 +0000242#endif // #ifndef _THRIFT_PROTOCOL_TBINARYPROTOCOL_H_