blob.h 8.3 KB
Newer Older
Kenton Varda's avatar
Kenton Varda committed
1 2
// Copyright (c) 2013-2014 Sandstorm Development Group, Inc. and contributors
// Licensed under the MIT License:
Kenton Varda's avatar
Kenton Varda committed
3
//
Kenton Varda's avatar
Kenton Varda committed
4 5 6 7 8 9
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
Kenton Varda's avatar
Kenton Varda committed
10
//
Kenton Varda's avatar
Kenton Varda committed
11 12
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
Kenton Varda's avatar
Kenton Varda committed
13
//
Kenton Varda's avatar
Kenton Varda committed
14 15 16 17 18 19 20
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
Kenton Varda's avatar
Kenton Varda committed
21

Kenton Varda's avatar
Kenton Varda committed
22 23
#ifndef CAPNP_BLOB_H_
#define CAPNP_BLOB_H_
Kenton Varda's avatar
Kenton Varda committed
24

25 26 27 28
#if defined(__GNUC__) && !CAPNP_HEADER_WARNINGS
#pragma GCC system_header
#endif

Kenton Varda's avatar
Kenton Varda committed
29
#include <kj/common.h>
Kenton Varda's avatar
Kenton Varda committed
30
#include <kj/string.h>
31
#include "common.h"
Kenton Varda's avatar
Kenton Varda committed
32 33
#include <string.h>

34
namespace capnp {
Kenton Varda's avatar
Kenton Varda committed
35 36

struct Data {
37
  Data() = delete;
Kenton Varda's avatar
Kenton Varda committed
38 39
  class Reader;
  class Builder;
40
  class Pipeline {};
Kenton Varda's avatar
Kenton Varda committed
41 42 43
};

struct Text {
44
  Text() = delete;
Kenton Varda's avatar
Kenton Varda committed
45 46
  class Reader;
  class Builder;
47
  class Pipeline {};
Kenton Varda's avatar
Kenton Varda committed
48 49
};

50
class Data::Reader: public kj::ArrayPtr<const byte> {
Kenton Varda's avatar
Kenton Varda committed
51 52 53 54
  // Points to a blob of bytes.  The usual Reader rules apply -- Data::Reader behaves like a simple
  // pointer which does not own its target, can be passed by value, etc.

public:
55 56
  typedef Data Reads;

57 58 59 60 61 62 63
  Reader() = default;
  inline Reader(decltype(nullptr)): ArrayPtr<const byte>(nullptr) {}
  inline Reader(const byte* value, size_t size): ArrayPtr<const byte>(value, size) {}
  inline Reader(const kj::Array<const byte>& value): ArrayPtr<const byte>(value) {}
  inline Reader(const ArrayPtr<const byte>& value): ArrayPtr<const byte>(value) {}
  inline Reader(const kj::Array<byte>& value): ArrayPtr<const byte>(value) {}
  inline Reader(const ArrayPtr<byte>& value): ArrayPtr<const byte>(value) {}
Kenton Varda's avatar
Kenton Varda committed
64 65
};

66
class Text::Reader: public kj::StringPtr {
Kenton Varda's avatar
Kenton Varda committed
67 68 69
  // Like Data::Reader, but points at NUL-terminated UTF-8 text.  The NUL terminator is not counted
  // in the size but must be present immediately after the last byte.
  //
70 71 72
  // Text::Reader's interface contract is that its data MUST be NUL-terminated.  The producer of
  // the Text::Reader must guarantee this, so that the consumer need not check.  The data SHOULD
  // also be valid UTF-8, but this is NOT guaranteed -- the consumer must verify if it cares.
Kenton Varda's avatar
Kenton Varda committed
73 74

public:
75 76
  typedef Text Reads;

77 78 79 80 81 82
  Reader() = default;
  inline Reader(decltype(nullptr)): StringPtr(nullptr) {}
  inline Reader(const char* value): StringPtr(value) {}
  inline Reader(const char* value, size_t size): StringPtr(value, size) {}
  inline Reader(const kj::String& value): StringPtr(value) {}
  inline Reader(const StringPtr& value): StringPtr(value) {}
83 84 85 86 87 88 89 90

#if KJ_COMPILER_SUPPORTS_STL_STRING_INTEROP
  template <typename T, typename = decltype(kj::instance<T>().c_str())>
  inline Reader(const T& t): StringPtr(t) {}
  // Allow implicit conversion from any class that has a c_str() method (namely, std::string).
  // We use a template trick to detect std::string in order to avoid including the header for
  // those who don't want it.
#endif
Kenton Varda's avatar
Kenton Varda committed
91 92
};

93 94
class Data::Builder: public kj::ArrayPtr<byte> {
  // Like Data::Reader except the pointers aren't const.
Kenton Varda's avatar
Kenton Varda committed
95 96

public:
97 98
  typedef Data Builds;

99 100 101 102
  Builder() = default;
  inline Builder(decltype(nullptr)): ArrayPtr<byte>(nullptr) {}
  inline Builder(byte* value, size_t size): ArrayPtr<byte>(value, size) {}
  inline Builder(kj::Array<byte>& value): ArrayPtr<byte>(value) {}
103
  inline Builder(ArrayPtr<byte> value): ArrayPtr<byte>(value) {}
Kenton Varda's avatar
Kenton Varda committed
104

105
  inline Data::Reader asReader() const { return Data::Reader(*this); }
106
  inline operator Reader() const { return asReader(); }
107
};
Kenton Varda's avatar
Kenton Varda committed
108

109
class Text::Builder: public kj::DisallowConstCopy {
110
  // Basically identical to kj::StringPtr, except that the contents are non-const.
Kenton Varda's avatar
Kenton Varda committed
111

112 113 114 115 116 117
public:
  inline Builder(): content(nulstr, 1) {}
  inline Builder(decltype(nullptr)): content(nulstr, 1) {}
  inline Builder(char* value): content(value, strlen(value) + 1) {}
  inline Builder(char* value, size_t size): content(value, size + 1) {
    KJ_IREQUIRE(value[size] == '\0', "StringPtr must be NUL-terminated.");
Kenton Varda's avatar
Kenton Varda committed
118 119
  }

120
  inline Reader asReader() const { return Reader(content.begin(), content.size() - 1); }
121
  inline operator Reader() const { return asReader(); }
Kenton Varda's avatar
Kenton Varda committed
122

123 124 125 126
  inline operator kj::ArrayPtr<char>();
  inline kj::ArrayPtr<char> asArray();
  inline operator kj::ArrayPtr<const char>() const;
  inline kj::ArrayPtr<const char> asArray() const;
127 128
  inline kj::ArrayPtr<byte> asBytes() { return asArray().asBytes(); }
  inline kj::ArrayPtr<const byte> asBytes() const { return asArray().asBytes(); }
129
  // Result does not include NUL terminator.
130

131 132
  inline operator kj::StringPtr() const;
  inline kj::StringPtr asString() const;
Kenton Varda's avatar
Kenton Varda committed
133

134 135
  inline const char* cStr() const { return content.begin(); }
  // Returns NUL-terminated string.
Kenton Varda's avatar
Kenton Varda committed
136

137 138
  inline size_t size() const { return content.size() - 1; }
  // Result does not include NUL terminator.
Kenton Varda's avatar
Kenton Varda committed
139

140
  inline char operator[](size_t index) const { return content[index]; }
141
  inline char& operator[](size_t index) { return content[index]; }
142

143 144 145 146
  inline char* begin() { return content.begin(); }
  inline char* end() { return content.end() - 1; }
  inline const char* begin() const { return content.begin(); }
  inline const char* end() const { return content.end() - 1; }
147

148 149
  inline bool operator==(decltype(nullptr)) const { return content.size() <= 1; }
  inline bool operator!=(decltype(nullptr)) const { return content.size() > 1; }
Kenton Varda's avatar
Kenton Varda committed
150

151 152 153 154 155 156
  inline bool operator==(Builder other) const { return asString() == other.asString(); }
  inline bool operator!=(Builder other) const { return asString() == other.asString(); }
  inline bool operator< (Builder other) const { return asString() <  other.asString(); }
  inline bool operator> (Builder other) const { return asString() >  other.asString(); }
  inline bool operator<=(Builder other) const { return asString() <= other.asString(); }
  inline bool operator>=(Builder other) const { return asString() >= other.asString(); }
Kenton Varda's avatar
Kenton Varda committed
157

158 159 160 161
  inline kj::StringPtr slice(size_t start) const;
  inline kj::ArrayPtr<const char> slice(size_t start, size_t end) const;
  inline Builder slice(size_t start);
  inline kj::ArrayPtr<char> slice(size_t start, size_t end);
162 163
  // A string slice is only NUL-terminated if it is a suffix, so slice() has a one-parameter
  // version that assumes end = size().
Kenton Varda's avatar
Kenton Varda committed
164

Kenton Varda's avatar
Kenton Varda committed
165
private:
166 167 168 169
  inline explicit Builder(kj::ArrayPtr<char> content): content(content) {}

  kj::ArrayPtr<char> content;

Kenton Varda's avatar
Kenton Varda committed
170 171 172
  static char nulstr[1];
};

Kenton Varda's avatar
Kenton Varda committed
173
inline kj::StringPtr KJ_STRINGIFY(Text::Builder builder) {
174
  return builder.asString();
Kenton Varda's avatar
Kenton Varda committed
175 176
}

177 178 179 180 181 182 183 184 185 186 187
inline bool operator==(const char* a, const Text::Builder& b) { return a == b.asString(); }
inline bool operator!=(const char* a, const Text::Builder& b) { return a != b.asString(); }

inline Text::Builder::operator kj::StringPtr() const {
  return kj::StringPtr(content.begin(), content.size() - 1);
}

inline kj::StringPtr Text::Builder::asString() const {
  return kj::StringPtr(content.begin(), content.size() - 1);
}

188
inline Text::Builder::operator kj::ArrayPtr<char>() {
189 190 191
  return content.slice(0, content.size() - 1);
}

192
inline kj::ArrayPtr<char> Text::Builder::asArray() {
193 194 195
  return content.slice(0, content.size() - 1);
}

196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211
inline Text::Builder::operator kj::ArrayPtr<const char>() const {
  return content.slice(0, content.size() - 1);
}

inline kj::ArrayPtr<const char> Text::Builder::asArray() const {
  return content.slice(0, content.size() - 1);
}

inline kj::StringPtr Text::Builder::slice(size_t start) const {
  return asReader().slice(start);
}
inline kj::ArrayPtr<const char> Text::Builder::slice(size_t start, size_t end) const {
  return content.slice(start, end);
}

inline Text::Builder Text::Builder::slice(size_t start) {
212 213
  return Text::Builder(content.slice(start, content.size()));
}
214
inline kj::ArrayPtr<char> Text::Builder::slice(size_t start, size_t end) {
215 216
  return content.slice(start, end);
}
217

218
}  // namespace capnp
Kenton Varda's avatar
Kenton Varda committed
219

Kenton Varda's avatar
Kenton Varda committed
220
#endif  // CAPNP_BLOB_H_