blob: dedeec35bce0a557a763005b519c62dcfcb89f75 [file] [log] [blame]
Nobuaki Sukegawa6525f6a2016-02-11 13:58:39 +09001/*
2 * Licensed to the Apache Software Foundation (ASF) under one
3 * or more contributor license agreements. See the NOTICE file
4 * distributed with this work for additional information
5 * regarding copyright ownership. The ASF licenses this file
6 * to you under the Apache License, Version 2.0 (the
7 * "License"); you may not use this file except in compliance
8 * with the License. You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing,
13 * software distributed under the License is distributed on an
14 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15 * KIND, either express or implied. See the License for the
16 * specific language governing permissions and limitations
17 * under the License.
18 */
19
20#ifndef THRIFT_PY_BINARY_H
21#define THRIFT_PY_BINARY_H
22
23#include <Python.h>
24#include "ext/protocol.h"
25#include "ext/endian.h"
26#include <stdint.h>
27
28namespace apache {
29namespace thrift {
30namespace py {
31
32class BinaryProtocol : public ProtocolBase<BinaryProtocol> {
33public:
34 virtual ~BinaryProtocol() {}
35
36 void writeI8(int8_t val) { writeBuffer(reinterpret_cast<char*>(&val), sizeof(int8_t)); }
37
38 void writeI16(int16_t val) {
39 int16_t net = static_cast<int16_t>(htons(val));
40 writeBuffer(reinterpret_cast<char*>(&net), sizeof(int16_t));
41 }
42
43 void writeI32(int32_t val) {
44 int32_t net = static_cast<int32_t>(htonl(val));
45 writeBuffer(reinterpret_cast<char*>(&net), sizeof(int32_t));
46 }
47
48 void writeI64(int64_t val) {
49 int64_t net = static_cast<int64_t>(htonll(val));
50 writeBuffer(reinterpret_cast<char*>(&net), sizeof(int64_t));
51 }
52
53 void writeDouble(double dub) {
54 // Unfortunately, bitwise_cast doesn't work in C. Bad C!
55 union {
56 double f;
57 int64_t t;
58 } transfer;
59 transfer.f = dub;
60 writeI64(transfer.t);
61 }
62
63 void writeBool(int v) { writeByte(static_cast<uint8_t>(v)); }
64
65 void writeString(PyObject* value, int32_t len) {
66 writeI32(len);
67 writeBuffer(PyBytes_AS_STRING(value), len);
68 }
69
70 bool writeListBegin(PyObject* value, const SetListTypeArgs& parsedargs, int32_t len) {
71 writeByte(parsedargs.element_type);
72 writeI32(len);
73 return true;
74 }
75
76 bool writeMapBegin(PyObject* value, const MapTypeArgs& parsedargs, int32_t len) {
77 writeByte(parsedargs.ktag);
78 writeByte(parsedargs.vtag);
79 writeI32(len);
80 return true;
81 }
82
83 bool writeStructBegin() { return true; }
84 bool writeStructEnd() { return true; }
85 bool writeField(PyObject* value, const StructItemSpec& parsedspec) {
86 writeByte(static_cast<uint8_t>(parsedspec.type));
87 writeI16(parsedspec.tag);
88 return encodeValue(value, parsedspec.type, parsedspec.typeargs);
89 }
90
91 void writeFieldStop() { writeByte(static_cast<uint8_t>(T_STOP)); }
92
93 bool readBool(bool& val) {
94 char* buf;
95 if (!readBytes(&buf, 1)) {
96 return false;
97 }
98 val = buf[0] == 1;
99 return true;
100 }
101
102 bool readI8(int8_t& val) {
103 char* buf;
104 if (!readBytes(&buf, 1)) {
105 return false;
106 }
107 val = buf[0];
108 return true;
109 }
110
111 bool readI16(int16_t& val) {
112 char* buf;
113 if (!readBytes(&buf, sizeof(int16_t))) {
114 return false;
115 }
116 val = static_cast<int16_t>(ntohs(*reinterpret_cast<int16_t*>(buf)));
117 return true;
118 }
119
120 bool readI32(int32_t& val) {
121 char* buf;
122 if (!readBytes(&buf, sizeof(int32_t))) {
123 return false;
124 }
125 val = static_cast<int32_t>(ntohl(*reinterpret_cast<int32_t*>(buf)));
126 return true;
127 }
128
129 bool readI64(int64_t& val) {
130 char* buf;
131 if (!readBytes(&buf, sizeof(int64_t))) {
132 return false;
133 }
134 val = static_cast<int64_t>(ntohll(*reinterpret_cast<int64_t*>(buf)));
135 return true;
136 }
137
138 bool readDouble(double& val) {
139 union {
140 int64_t f;
141 double t;
142 } transfer;
143
144 if (!readI64(transfer.f)) {
145 return false;
146 }
147 val = transfer.t;
148 return true;
149 }
150
151 int32_t readString(char** buf) {
152 int32_t len = 0;
153 if (!readI32(len) || !checkLengthLimit(len, stringLimit()) || !readBytes(buf, len)) {
154 return -1;
155 }
156 return len;
157 }
158
159 int32_t readListBegin(TType& etype) {
160 int32_t len;
161 uint8_t b = 0;
162 if (!readByte(b) || !readI32(len) || !checkLengthLimit(len, containerLimit())) {
163 return -1;
164 }
165 etype = static_cast<TType>(b);
166 return len;
167 }
168
169 int32_t readMapBegin(TType& ktype, TType& vtype) {
170 int32_t len;
171 uint8_t k, v;
172 if (!readByte(k) || !readByte(v) || !readI32(len) || !checkLengthLimit(len, containerLimit())) {
173 return -1;
174 }
175 ktype = static_cast<TType>(k);
176 vtype = static_cast<TType>(v);
177 return len;
178 }
179
180 bool readStructBegin() { return true; }
181 bool readStructEnd() { return true; }
182
183 bool readFieldBegin(TType& type, int16_t& tag);
184
185#define SKIPBYTES(n) \
186 do { \
187 if (!readBytes(&dummy_buf_, (n))) { \
188 return false; \
189 } \
190 return true; \
191 } while (0)
192
193 bool skipBool() { SKIPBYTES(1); }
194 bool skipByte() { SKIPBYTES(1); }
195 bool skipI16() { SKIPBYTES(2); }
196 bool skipI32() { SKIPBYTES(4); }
197 bool skipI64() { SKIPBYTES(8); }
198 bool skipDouble() { SKIPBYTES(8); }
199 bool skipString() {
200 int32_t len;
201 if (!readI32(len)) {
202 return false;
203 }
204 SKIPBYTES(len);
205 }
206#undef SKIPBYTES
207
208private:
209 char* dummy_buf_;
210};
211}
212}
213}
214#endif // THRIFT_PY_BINARY_H