blob.h 7.74 KB
Newer Older
Kenton Varda's avatar
Kenton Varda committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
// Copyright (c) 2013, Kenton Varda <temporal@gmail.com>
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this
//    list of conditions and the following disclaimer.
// 2. Redistributions in binary form must reproduce the above copyright notice,
//    this list of conditions and the following disclaimer in the documentation
//    and/or other materials provided with the distribution.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
// ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
// (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
// LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
// ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

Kenton Varda's avatar
Kenton Varda committed
24 25
#ifndef CAPNP_BLOB_H_
#define CAPNP_BLOB_H_
Kenton Varda's avatar
Kenton Varda committed
26

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

32
namespace capnp {
Kenton Varda's avatar
Kenton Varda committed
33 34

struct Data {
35
  Data() = delete;
Kenton Varda's avatar
Kenton Varda committed
36 37 38 39 40
  class Reader;
  class Builder;
};

struct Text {
41
  Text() = delete;
Kenton Varda's avatar
Kenton Varda committed
42 43 44 45
  class Reader;
  class Builder;
};

46
class Data::Reader: public kj::ArrayPtr<const byte> {
Kenton Varda's avatar
Kenton Varda committed
47 48 49 50
  // 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:
51 52
  typedef Data Reads;

53 54 55 56 57 58 59
  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
60 61
};

62
class Text::Reader: public kj::StringPtr {
Kenton Varda's avatar
Kenton Varda committed
63 64 65
  // 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.
  //
66 67 68
  // 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
69 70

public:
71 72
  typedef Text Reads;

73 74 75 76 77 78
  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) {}
Kenton Varda's avatar
Kenton Varda committed
79 80
};

81 82
class Data::Builder: public kj::ArrayPtr<byte> {
  // Like Data::Reader except the pointers aren't const.
Kenton Varda's avatar
Kenton Varda committed
83 84

public:
85 86
  typedef Data Builds;

87 88 89 90
  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) {}
91
  inline Builder(ArrayPtr<byte>& value): ArrayPtr<byte>(value) {}
Kenton Varda's avatar
Kenton Varda committed
92

93 94
  inline Data::Reader asReader() const { return Data::Reader(*this); }
};
Kenton Varda's avatar
Kenton Varda committed
95

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

99 100 101 102 103 104
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
105 106
  }

107
  inline Reader asReader() const { return Reader(content.begin(), content.size() - 1); }
Kenton Varda's avatar
Kenton Varda committed
108

109 110 111 112
  inline operator kj::ArrayPtr<char>();
  inline kj::ArrayPtr<char> asArray();
  inline operator kj::ArrayPtr<const char>() const;
  inline kj::ArrayPtr<const char> asArray() const;
113
  // Result does not include NUL terminator.
114

115 116
  inline operator kj::StringPtr() const;
  inline kj::StringPtr asString() const;
Kenton Varda's avatar
Kenton Varda committed
117

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

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

124
  inline char operator[](size_t index) const { return content[index]; }
125
  inline char& operator[](size_t index) { return content[index]; }
126

127 128 129 130
  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; }
131

132 133
  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
134

135 136 137 138 139 140
  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
141

142 143 144 145
  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);
146 147
  // 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
148

Kenton Varda's avatar
Kenton Varda committed
149
private:
150 151 152 153
  inline explicit Builder(kj::ArrayPtr<char> content): content(content) {}

  kj::ArrayPtr<char> content;

Kenton Varda's avatar
Kenton Varda committed
154 155 156
  static char nulstr[1];
};

Kenton Varda's avatar
Kenton Varda committed
157
inline kj::StringPtr KJ_STRINGIFY(Text::Builder builder) {
158
  return builder.asString();
Kenton Varda's avatar
Kenton Varda committed
159 160
}

161 162 163 164 165 166 167 168 169 170 171
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);
}

172
inline Text::Builder::operator kj::ArrayPtr<char>() {
173 174 175
  return content.slice(0, content.size() - 1);
}

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

180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195
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) {
196 197
  return Text::Builder(content.slice(start, content.size()));
}
198
inline kj::ArrayPtr<char> Text::Builder::slice(size_t start, size_t end) {
199 200
  return content.slice(start, end);
}
201

202
}  // namespace capnp
Kenton Varda's avatar
Kenton Varda committed
203

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