mirror of https://github.com/axmolengine/axmol.git
225 lines
5.8 KiB
C++
225 lines
5.8 KiB
C++
|
//////////////////////////////////////////////////////////////////////////////////////////
|
||
|
// A cross platform socket APIs, support ios & android & wp8 & window store
|
||
|
// universal app
|
||
|
//////////////////////////////////////////////////////////////////////////////////////////
|
||
|
/*
|
||
|
The MIT License (MIT)
|
||
|
|
||
|
Copyright (c) 2012-2020 HALX99
|
||
|
|
||
|
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:
|
||
|
|
||
|
The above copyright notice and this permission notice shall be included in all
|
||
|
copies or substantial portions of the Software.
|
||
|
|
||
|
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.
|
||
|
*/
|
||
|
#ifndef YASIO__OBSTREAM_CPP
|
||
|
#define YASIO__OBSTREAM_CPP
|
||
|
#if !defined(YASIO_HEADER_ONLY)
|
||
|
# include "yasio/obstream.hpp"
|
||
|
#endif
|
||
|
#include <iostream>
|
||
|
#include <fstream>
|
||
|
|
||
|
namespace yasio
|
||
|
{
|
||
|
|
||
|
obstream::obstream(size_t capacity) { buffer_.reserve(capacity); }
|
||
|
|
||
|
void obstream::push8()
|
||
|
{
|
||
|
offset_stack_.push(buffer_.size());
|
||
|
write_i(static_cast<uint8_t>(0));
|
||
|
}
|
||
|
void obstream::pop8()
|
||
|
{
|
||
|
auto offset = offset_stack_.top();
|
||
|
pwrite_ix(offset, static_cast<uint8_t>(buffer_.size() - offset - sizeof(uint8_t)));
|
||
|
offset_stack_.pop();
|
||
|
}
|
||
|
void obstream::pop8(uint8_t value)
|
||
|
{
|
||
|
auto offset = offset_stack_.top();
|
||
|
pwrite_ix(offset, value);
|
||
|
offset_stack_.pop();
|
||
|
}
|
||
|
|
||
|
void obstream::push16()
|
||
|
{
|
||
|
offset_stack_.push(buffer_.size());
|
||
|
write_i(static_cast<uint16_t>(0));
|
||
|
}
|
||
|
void obstream::pop16()
|
||
|
{
|
||
|
auto offset = offset_stack_.top();
|
||
|
pwrite_ix(offset, static_cast<uint16_t>(buffer_.size() - offset - sizeof(uint16_t)));
|
||
|
offset_stack_.pop();
|
||
|
}
|
||
|
void obstream::pop16(uint16_t value)
|
||
|
{
|
||
|
auto offset = offset_stack_.top();
|
||
|
pwrite_ix(offset, value);
|
||
|
offset_stack_.pop();
|
||
|
}
|
||
|
|
||
|
void obstream::push24()
|
||
|
{
|
||
|
offset_stack_.push(buffer_.size());
|
||
|
|
||
|
unsigned char u32buf[4] = {0, 0, 0, 0};
|
||
|
write_bytes(u32buf, 3);
|
||
|
}
|
||
|
void obstream::pop24()
|
||
|
{
|
||
|
auto offset = offset_stack_.top();
|
||
|
auto value = htonl(static_cast<uint32_t>(buffer_.size() - offset - 3)) >> 8;
|
||
|
memcpy(wptr(offset), &value, 3);
|
||
|
offset_stack_.pop();
|
||
|
}
|
||
|
void obstream::pop24(uint32_t value)
|
||
|
{
|
||
|
auto offset = offset_stack_.top();
|
||
|
value = htonl(value) >> 8;
|
||
|
memcpy(wptr(offset), &value, 3);
|
||
|
offset_stack_.pop();
|
||
|
}
|
||
|
|
||
|
void obstream::push32()
|
||
|
{
|
||
|
offset_stack_.push(buffer_.size());
|
||
|
write_i(static_cast<uint32_t>(0));
|
||
|
}
|
||
|
|
||
|
void obstream::pop32()
|
||
|
{
|
||
|
auto offset = offset_stack_.top();
|
||
|
pwrite_ix(offset, static_cast<uint32_t>(buffer_.size() - offset - sizeof(uint32_t)));
|
||
|
offset_stack_.pop();
|
||
|
}
|
||
|
|
||
|
void obstream::pop32(uint32_t value)
|
||
|
{
|
||
|
auto offset = offset_stack_.top();
|
||
|
pwrite_ix(offset, value);
|
||
|
offset_stack_.pop();
|
||
|
}
|
||
|
|
||
|
obstream::obstream(const obstream& right) : buffer_(right.buffer_) {}
|
||
|
|
||
|
obstream::obstream(obstream&& right) : buffer_(std::move(right.buffer_)) {}
|
||
|
|
||
|
obstream::~obstream() {}
|
||
|
|
||
|
obstream& obstream::operator=(const obstream& right)
|
||
|
{
|
||
|
buffer_ = right.buffer_;
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
obstream& obstream::operator=(obstream&& right)
|
||
|
{
|
||
|
buffer_ = std::move(right.buffer_);
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
void obstream::write_i24(int32_t value) { write_u24(value); }
|
||
|
|
||
|
void obstream::write_u24(uint32_t value)
|
||
|
{
|
||
|
value = htonl(value) >> 8;
|
||
|
write_bytes(&value, 3);
|
||
|
}
|
||
|
|
||
|
void obstream::write_i(int value)
|
||
|
{
|
||
|
// Write out an int 7 bits at a time. The high bit of the byte,
|
||
|
// when on, tells reader to continue reading more bytes.
|
||
|
uint32_t v = (uint32_t)value; // support negative numbers
|
||
|
while (v >= 0x80)
|
||
|
{
|
||
|
write_byte((uint8_t)(v | 0x80));
|
||
|
v >>= 7;
|
||
|
}
|
||
|
write_byte((uint8_t)v);
|
||
|
}
|
||
|
|
||
|
void obstream::write_v(cxx17::string_view sv)
|
||
|
{
|
||
|
int len = static_cast<int>(sv.length());
|
||
|
write_i(len);
|
||
|
write_bytes(sv.data(), len);
|
||
|
}
|
||
|
|
||
|
void obstream::write_v32(cxx17::string_view value)
|
||
|
{
|
||
|
write_v32(value.data(), static_cast<int>(value.size()));
|
||
|
}
|
||
|
void obstream::write_v16(cxx17::string_view value)
|
||
|
{
|
||
|
write_v16(value.data(), static_cast<int>(value.size()));
|
||
|
}
|
||
|
void obstream::write_v8(cxx17::string_view value)
|
||
|
{
|
||
|
write_v8(value.data(), static_cast<int>(value.size()));
|
||
|
}
|
||
|
|
||
|
void obstream::write_v32(const void* v, int size) { write_vx<uint32_t>(v, size); }
|
||
|
void obstream::write_v16(const void* v, int size) { write_vx<uint16_t>(v, size); }
|
||
|
void obstream::write_v8(const void* v, int size) { write_vx<uint8_t>(v, size); }
|
||
|
|
||
|
void obstream::write_byte(uint8_t v) { buffer_.push_back(v); }
|
||
|
|
||
|
void obstream::write_bytes(cxx17::string_view v)
|
||
|
{
|
||
|
return write_bytes(v.data(), static_cast<int>(v.size()));
|
||
|
}
|
||
|
void obstream::write_bytes(const void* v, int vl)
|
||
|
{
|
||
|
if (vl > 0)
|
||
|
buffer_.insert(buffer_.end(), (const char*)v, (const char*)v + vl);
|
||
|
}
|
||
|
void obstream::write_bytes(std::streamoff offset, const void* v, int vl)
|
||
|
{
|
||
|
if ((offset + vl) < static_cast<std::streamoff>(buffer_.size()))
|
||
|
::memcpy(buffer_.data() + offset, v, vl);
|
||
|
}
|
||
|
|
||
|
void obstream::save(const char* filename)
|
||
|
{
|
||
|
std::ofstream fout;
|
||
|
fout.open(filename, std::ios::binary);
|
||
|
if (!this->buffer_.empty())
|
||
|
fout.write(&this->buffer_.front(), this->length());
|
||
|
fout.close();
|
||
|
}
|
||
|
|
||
|
obstream obstream::sub(size_t offset, size_t count)
|
||
|
{
|
||
|
obstream obs;
|
||
|
auto n = length();
|
||
|
if (offset < n)
|
||
|
{
|
||
|
if (count > (n - offset))
|
||
|
count = (n - offset);
|
||
|
|
||
|
obs.buffer_.assign(this->data() + offset, this->data() + offset + count);
|
||
|
}
|
||
|
return obs;
|
||
|
}
|
||
|
|
||
|
} // namespace yasio
|
||
|
|
||
|
#endif
|