175 lines
5.3 KiB
C++
175 lines
5.3 KiB
C++
|
/*
|
||
|
* Tencent is pleased to support the open source community by making
|
||
|
* MMKV available.
|
||
|
*
|
||
|
* Copyright (C) 2018 THL A29 Limited, a Tencent company.
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* Licensed under the BSD 3-Clause License (the "License"); you may not use
|
||
|
* this file except in compliance with the License. You may obtain a copy of
|
||
|
* the License at
|
||
|
*
|
||
|
* https://opensource.org/licenses/BSD-3-Clause
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
* See the License for the specific language governing permissions and
|
||
|
* limitations under the License.
|
||
|
*/
|
||
|
|
||
|
#include "CodedOutputData.h"
|
||
|
#include "PBUtility.h"
|
||
|
#include <cstring>
|
||
|
#include <stdexcept>
|
||
|
|
||
|
#ifdef MMKV_APPLE
|
||
|
# if __has_feature(objc_arc)
|
||
|
# error This file must be compiled with MRC. Use -fno-objc-arc flag.
|
||
|
# endif
|
||
|
#endif // MMKV_APPLE
|
||
|
|
||
|
using namespace std;
|
||
|
|
||
|
namespace mmkv {
|
||
|
|
||
|
CodedOutputData::CodedOutputData(void *ptr, size_t len) : m_ptr((uint8_t *) ptr), m_size(len), m_position(0) {
|
||
|
MMKV_ASSERT(m_ptr);
|
||
|
}
|
||
|
|
||
|
uint8_t *CodedOutputData::curWritePointer() {
|
||
|
return m_ptr + m_position;
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeDouble(double value) {
|
||
|
this->writeRawLittleEndian64(Float64ToInt64(value));
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeFloat(float value) {
|
||
|
this->writeRawLittleEndian32(Float32ToInt32(value));
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeInt64(int64_t value) {
|
||
|
this->writeRawVarint64(value);
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeUInt64(uint64_t value) {
|
||
|
writeRawVarint64(static_cast<int64_t>(value));
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeInt32(int32_t value) {
|
||
|
if (value >= 0) {
|
||
|
this->writeRawVarint32(value);
|
||
|
} else {
|
||
|
this->writeRawVarint64(value);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeUInt32(uint32_t value) {
|
||
|
writeRawVarint32(static_cast<int32_t>(value));
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeBool(bool value) {
|
||
|
this->writeRawByte(static_cast<uint8_t>(value ? 1 : 0));
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeData(const MMBuffer &value) {
|
||
|
this->writeRawVarint32((int32_t) value.length());
|
||
|
this->writeRawData(value);
|
||
|
}
|
||
|
|
||
|
#ifndef MMKV_APPLE
|
||
|
|
||
|
void CodedOutputData::writeString(const string &value) {
|
||
|
size_t numberOfBytes = value.size();
|
||
|
this->writeRawVarint32((int32_t) numberOfBytes);
|
||
|
if (m_position + numberOfBytes > m_size) {
|
||
|
auto msg = "m_position: " + to_string(m_position) + ", numberOfBytes: " + to_string(numberOfBytes) +
|
||
|
", m_size: " + to_string(m_size);
|
||
|
throw out_of_range(msg);
|
||
|
}
|
||
|
memcpy(m_ptr + m_position, ((uint8_t *) value.data()), numberOfBytes);
|
||
|
m_position += numberOfBytes;
|
||
|
}
|
||
|
|
||
|
#endif // MMKV_APPLE
|
||
|
|
||
|
size_t CodedOutputData::spaceLeft() {
|
||
|
if (m_size <= m_position) {
|
||
|
return 0;
|
||
|
}
|
||
|
return m_size - m_position;
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::seek(size_t addedSize) {
|
||
|
m_position += addedSize;
|
||
|
|
||
|
if (m_position > m_size) {
|
||
|
throw out_of_range("OutOfSpace");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeRawByte(uint8_t value) {
|
||
|
if (m_position == m_size) {
|
||
|
throw out_of_range("m_position: " + to_string(m_position) + " m_size: " + to_string(m_size));
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
m_ptr[m_position++] = value;
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeRawData(const MMBuffer &data) {
|
||
|
size_t numberOfBytes = data.length();
|
||
|
if (m_position + numberOfBytes > m_size) {
|
||
|
auto msg = "m_position: " + to_string(m_position) + ", numberOfBytes: " + to_string(numberOfBytes) +
|
||
|
", m_size: " + to_string(m_size);
|
||
|
throw out_of_range(msg);
|
||
|
}
|
||
|
memcpy(m_ptr + m_position, data.getPtr(), numberOfBytes);
|
||
|
m_position += numberOfBytes;
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeRawVarint32(int32_t value) {
|
||
|
while (true) {
|
||
|
if ((value & ~0x7f) == 0) {
|
||
|
this->writeRawByte(static_cast<uint8_t>(value));
|
||
|
return;
|
||
|
} else {
|
||
|
this->writeRawByte(static_cast<uint8_t>((value & 0x7F) | 0x80));
|
||
|
value = logicalRightShift32(value, 7);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeRawVarint64(int64_t value) {
|
||
|
while (true) {
|
||
|
if ((value & ~0x7f) == 0) {
|
||
|
this->writeRawByte(static_cast<uint8_t>(value));
|
||
|
return;
|
||
|
} else {
|
||
|
this->writeRawByte(static_cast<uint8_t>((value & 0x7f) | 0x80));
|
||
|
value = logicalRightShift64(value, 7);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeRawLittleEndian32(int32_t value) {
|
||
|
this->writeRawByte(static_cast<uint8_t>((value) &0xff));
|
||
|
this->writeRawByte(static_cast<uint8_t>((value >> 8) & 0xff));
|
||
|
this->writeRawByte(static_cast<uint8_t>((value >> 16) & 0xff));
|
||
|
this->writeRawByte(static_cast<uint8_t>((value >> 24) & 0xff));
|
||
|
}
|
||
|
|
||
|
void CodedOutputData::writeRawLittleEndian64(int64_t value) {
|
||
|
this->writeRawByte(static_cast<uint8_t>((value) &0xff));
|
||
|
this->writeRawByte(static_cast<uint8_t>((value >> 8) & 0xff));
|
||
|
this->writeRawByte(static_cast<uint8_t>((value >> 16) & 0xff));
|
||
|
this->writeRawByte(static_cast<uint8_t>((value >> 24) & 0xff));
|
||
|
this->writeRawByte(static_cast<uint8_t>((value >> 32) & 0xff));
|
||
|
this->writeRawByte(static_cast<uint8_t>((value >> 40) & 0xff));
|
||
|
this->writeRawByte(static_cast<uint8_t>((value >> 48) & 0xff));
|
||
|
this->writeRawByte(static_cast<uint8_t>((value >> 56) & 0xff));
|
||
|
}
|
||
|
|
||
|
} // namespace mmkv
|