cannam@147
|
1 // Copyright (c) 2013-2014 Sandstorm Development Group, Inc. and contributors
|
cannam@147
|
2 // Licensed under the MIT License:
|
cannam@147
|
3 //
|
cannam@147
|
4 // Permission is hereby granted, free of charge, to any person obtaining a copy
|
cannam@147
|
5 // of this software and associated documentation files (the "Software"), to deal
|
cannam@147
|
6 // in the Software without restriction, including without limitation the rights
|
cannam@147
|
7 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
cannam@147
|
8 // copies of the Software, and to permit persons to whom the Software is
|
cannam@147
|
9 // furnished to do so, subject to the following conditions:
|
cannam@147
|
10 //
|
cannam@147
|
11 // The above copyright notice and this permission notice shall be included in
|
cannam@147
|
12 // all copies or substantial portions of the Software.
|
cannam@147
|
13 //
|
cannam@147
|
14 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
cannam@147
|
15 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
cannam@147
|
16 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
cannam@147
|
17 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
cannam@147
|
18 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
cannam@147
|
19 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
cannam@147
|
20 // THE SOFTWARE.
|
cannam@147
|
21
|
cannam@147
|
22 #ifndef CAPNP_BLOB_H_
|
cannam@147
|
23 #define CAPNP_BLOB_H_
|
cannam@147
|
24
|
cannam@147
|
25 #if defined(__GNUC__) && !defined(CAPNP_HEADER_WARNINGS)
|
cannam@147
|
26 #pragma GCC system_header
|
cannam@147
|
27 #endif
|
cannam@147
|
28
|
cannam@147
|
29 #include <kj/common.h>
|
cannam@147
|
30 #include <kj/string.h>
|
cannam@147
|
31 #include "common.h"
|
cannam@147
|
32 #include <string.h>
|
cannam@147
|
33
|
cannam@147
|
34 namespace capnp {
|
cannam@147
|
35
|
cannam@147
|
36 struct Data {
|
cannam@147
|
37 Data() = delete;
|
cannam@147
|
38 class Reader;
|
cannam@147
|
39 class Builder;
|
cannam@147
|
40 class Pipeline {};
|
cannam@147
|
41 };
|
cannam@147
|
42
|
cannam@147
|
43 struct Text {
|
cannam@147
|
44 Text() = delete;
|
cannam@147
|
45 class Reader;
|
cannam@147
|
46 class Builder;
|
cannam@147
|
47 class Pipeline {};
|
cannam@147
|
48 };
|
cannam@147
|
49
|
cannam@147
|
50 class Data::Reader: public kj::ArrayPtr<const byte> {
|
cannam@147
|
51 // Points to a blob of bytes. The usual Reader rules apply -- Data::Reader behaves like a simple
|
cannam@147
|
52 // pointer which does not own its target, can be passed by value, etc.
|
cannam@147
|
53
|
cannam@147
|
54 public:
|
cannam@147
|
55 typedef Data Reads;
|
cannam@147
|
56
|
cannam@147
|
57 Reader() = default;
|
cannam@147
|
58 inline Reader(decltype(nullptr)): ArrayPtr<const byte>(nullptr) {}
|
cannam@147
|
59 inline Reader(const byte* value, size_t size): ArrayPtr<const byte>(value, size) {}
|
cannam@147
|
60 inline Reader(const kj::Array<const byte>& value): ArrayPtr<const byte>(value) {}
|
cannam@147
|
61 inline Reader(const ArrayPtr<const byte>& value): ArrayPtr<const byte>(value) {}
|
cannam@147
|
62 inline Reader(const kj::Array<byte>& value): ArrayPtr<const byte>(value) {}
|
cannam@147
|
63 inline Reader(const ArrayPtr<byte>& value): ArrayPtr<const byte>(value) {}
|
cannam@147
|
64 };
|
cannam@147
|
65
|
cannam@147
|
66 class Text::Reader: public kj::StringPtr {
|
cannam@147
|
67 // Like Data::Reader, but points at NUL-terminated UTF-8 text. The NUL terminator is not counted
|
cannam@147
|
68 // in the size but must be present immediately after the last byte.
|
cannam@147
|
69 //
|
cannam@147
|
70 // Text::Reader's interface contract is that its data MUST be NUL-terminated. The producer of
|
cannam@147
|
71 // the Text::Reader must guarantee this, so that the consumer need not check. The data SHOULD
|
cannam@147
|
72 // also be valid UTF-8, but this is NOT guaranteed -- the consumer must verify if it cares.
|
cannam@147
|
73
|
cannam@147
|
74 public:
|
cannam@147
|
75 typedef Text Reads;
|
cannam@147
|
76
|
cannam@147
|
77 Reader() = default;
|
cannam@147
|
78 inline Reader(decltype(nullptr)): StringPtr(nullptr) {}
|
cannam@147
|
79 inline Reader(const char* value): StringPtr(value) {}
|
cannam@147
|
80 inline Reader(const char* value, size_t size): StringPtr(value, size) {}
|
cannam@147
|
81 inline Reader(const kj::String& value): StringPtr(value) {}
|
cannam@147
|
82 inline Reader(const StringPtr& value): StringPtr(value) {}
|
cannam@147
|
83
|
cannam@147
|
84 #if KJ_COMPILER_SUPPORTS_STL_STRING_INTEROP
|
cannam@147
|
85 template <typename T, typename = decltype(kj::instance<T>().c_str())>
|
cannam@147
|
86 inline Reader(const T& t): StringPtr(t) {}
|
cannam@147
|
87 // Allow implicit conversion from any class that has a c_str() method (namely, std::string).
|
cannam@147
|
88 // We use a template trick to detect std::string in order to avoid including the header for
|
cannam@147
|
89 // those who don't want it.
|
cannam@147
|
90 #endif
|
cannam@147
|
91 };
|
cannam@147
|
92
|
cannam@147
|
93 class Data::Builder: public kj::ArrayPtr<byte> {
|
cannam@147
|
94 // Like Data::Reader except the pointers aren't const.
|
cannam@147
|
95
|
cannam@147
|
96 public:
|
cannam@147
|
97 typedef Data Builds;
|
cannam@147
|
98
|
cannam@147
|
99 Builder() = default;
|
cannam@147
|
100 inline Builder(decltype(nullptr)): ArrayPtr<byte>(nullptr) {}
|
cannam@147
|
101 inline Builder(byte* value, size_t size): ArrayPtr<byte>(value, size) {}
|
cannam@147
|
102 inline Builder(kj::Array<byte>& value): ArrayPtr<byte>(value) {}
|
cannam@147
|
103 inline Builder(ArrayPtr<byte> value): ArrayPtr<byte>(value) {}
|
cannam@147
|
104
|
cannam@147
|
105 inline Data::Reader asReader() const { return Data::Reader(*this); }
|
cannam@147
|
106 inline operator Reader() const { return asReader(); }
|
cannam@147
|
107 };
|
cannam@147
|
108
|
cannam@147
|
109 class Text::Builder: public kj::DisallowConstCopy {
|
cannam@147
|
110 // Basically identical to kj::StringPtr, except that the contents are non-const.
|
cannam@147
|
111
|
cannam@147
|
112 public:
|
cannam@147
|
113 inline Builder(): content(nulstr, 1) {}
|
cannam@147
|
114 inline Builder(decltype(nullptr)): content(nulstr, 1) {}
|
cannam@147
|
115 inline Builder(char* value): content(value, strlen(value) + 1) {}
|
cannam@147
|
116 inline Builder(char* value, size_t size): content(value, size + 1) {
|
cannam@147
|
117 KJ_IREQUIRE(value[size] == '\0', "StringPtr must be NUL-terminated.");
|
cannam@147
|
118 }
|
cannam@147
|
119
|
cannam@147
|
120 inline Reader asReader() const { return Reader(content.begin(), content.size() - 1); }
|
cannam@147
|
121 inline operator Reader() const { return asReader(); }
|
cannam@147
|
122
|
cannam@147
|
123 inline operator kj::ArrayPtr<char>();
|
cannam@147
|
124 inline kj::ArrayPtr<char> asArray();
|
cannam@147
|
125 inline operator kj::ArrayPtr<const char>() const;
|
cannam@147
|
126 inline kj::ArrayPtr<const char> asArray() const;
|
cannam@147
|
127 inline kj::ArrayPtr<byte> asBytes() { return asArray().asBytes(); }
|
cannam@147
|
128 inline kj::ArrayPtr<const byte> asBytes() const { return asArray().asBytes(); }
|
cannam@147
|
129 // Result does not include NUL terminator.
|
cannam@147
|
130
|
cannam@147
|
131 inline operator kj::StringPtr() const;
|
cannam@147
|
132 inline kj::StringPtr asString() const;
|
cannam@147
|
133
|
cannam@147
|
134 inline const char* cStr() const { return content.begin(); }
|
cannam@147
|
135 // Returns NUL-terminated string.
|
cannam@147
|
136
|
cannam@147
|
137 inline size_t size() const { return content.size() - 1; }
|
cannam@147
|
138 // Result does not include NUL terminator.
|
cannam@147
|
139
|
cannam@147
|
140 inline char operator[](size_t index) const { return content[index]; }
|
cannam@147
|
141 inline char& operator[](size_t index) { return content[index]; }
|
cannam@147
|
142
|
cannam@147
|
143 inline char* begin() { return content.begin(); }
|
cannam@147
|
144 inline char* end() { return content.end() - 1; }
|
cannam@147
|
145 inline const char* begin() const { return content.begin(); }
|
cannam@147
|
146 inline const char* end() const { return content.end() - 1; }
|
cannam@147
|
147
|
cannam@147
|
148 inline bool operator==(decltype(nullptr)) const { return content.size() <= 1; }
|
cannam@147
|
149 inline bool operator!=(decltype(nullptr)) const { return content.size() > 1; }
|
cannam@147
|
150
|
cannam@147
|
151 inline bool operator==(Builder other) const { return asString() == other.asString(); }
|
cannam@147
|
152 inline bool operator!=(Builder other) const { return asString() != other.asString(); }
|
cannam@147
|
153 inline bool operator< (Builder other) const { return asString() < other.asString(); }
|
cannam@147
|
154 inline bool operator> (Builder other) const { return asString() > other.asString(); }
|
cannam@147
|
155 inline bool operator<=(Builder other) const { return asString() <= other.asString(); }
|
cannam@147
|
156 inline bool operator>=(Builder other) const { return asString() >= other.asString(); }
|
cannam@147
|
157
|
cannam@147
|
158 inline kj::StringPtr slice(size_t start) const;
|
cannam@147
|
159 inline kj::ArrayPtr<const char> slice(size_t start, size_t end) const;
|
cannam@147
|
160 inline Builder slice(size_t start);
|
cannam@147
|
161 inline kj::ArrayPtr<char> slice(size_t start, size_t end);
|
cannam@147
|
162 // A string slice is only NUL-terminated if it is a suffix, so slice() has a one-parameter
|
cannam@147
|
163 // version that assumes end = size().
|
cannam@147
|
164
|
cannam@147
|
165 private:
|
cannam@147
|
166 inline explicit Builder(kj::ArrayPtr<char> content): content(content) {}
|
cannam@147
|
167
|
cannam@147
|
168 kj::ArrayPtr<char> content;
|
cannam@147
|
169
|
cannam@147
|
170 static char nulstr[1];
|
cannam@147
|
171 };
|
cannam@147
|
172
|
cannam@147
|
173 inline kj::StringPtr KJ_STRINGIFY(Text::Builder builder) {
|
cannam@147
|
174 return builder.asString();
|
cannam@147
|
175 }
|
cannam@147
|
176
|
cannam@147
|
177 inline bool operator==(const char* a, const Text::Builder& b) { return a == b.asString(); }
|
cannam@147
|
178 inline bool operator!=(const char* a, const Text::Builder& b) { return a != b.asString(); }
|
cannam@147
|
179
|
cannam@147
|
180 inline Text::Builder::operator kj::StringPtr() const {
|
cannam@147
|
181 return kj::StringPtr(content.begin(), content.size() - 1);
|
cannam@147
|
182 }
|
cannam@147
|
183
|
cannam@147
|
184 inline kj::StringPtr Text::Builder::asString() const {
|
cannam@147
|
185 return kj::StringPtr(content.begin(), content.size() - 1);
|
cannam@147
|
186 }
|
cannam@147
|
187
|
cannam@147
|
188 inline Text::Builder::operator kj::ArrayPtr<char>() {
|
cannam@147
|
189 return content.slice(0, content.size() - 1);
|
cannam@147
|
190 }
|
cannam@147
|
191
|
cannam@147
|
192 inline kj::ArrayPtr<char> Text::Builder::asArray() {
|
cannam@147
|
193 return content.slice(0, content.size() - 1);
|
cannam@147
|
194 }
|
cannam@147
|
195
|
cannam@147
|
196 inline Text::Builder::operator kj::ArrayPtr<const char>() const {
|
cannam@147
|
197 return content.slice(0, content.size() - 1);
|
cannam@147
|
198 }
|
cannam@147
|
199
|
cannam@147
|
200 inline kj::ArrayPtr<const char> Text::Builder::asArray() const {
|
cannam@147
|
201 return content.slice(0, content.size() - 1);
|
cannam@147
|
202 }
|
cannam@147
|
203
|
cannam@147
|
204 inline kj::StringPtr Text::Builder::slice(size_t start) const {
|
cannam@147
|
205 return asReader().slice(start);
|
cannam@147
|
206 }
|
cannam@147
|
207 inline kj::ArrayPtr<const char> Text::Builder::slice(size_t start, size_t end) const {
|
cannam@147
|
208 return content.slice(start, end);
|
cannam@147
|
209 }
|
cannam@147
|
210
|
cannam@147
|
211 inline Text::Builder Text::Builder::slice(size_t start) {
|
cannam@147
|
212 return Text::Builder(content.slice(start, content.size()));
|
cannam@147
|
213 }
|
cannam@147
|
214 inline kj::ArrayPtr<char> Text::Builder::slice(size_t start, size_t end) {
|
cannam@147
|
215 return content.slice(start, end);
|
cannam@147
|
216 }
|
cannam@147
|
217
|
cannam@147
|
218 } // namespace capnp
|
cannam@147
|
219
|
cannam@147
|
220 #endif // CAPNP_BLOB_H_
|