/*
* Copyright (C) 2016 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* 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.
*/
#ifndef _MESSAGE_BUF_H_
#define _MESSAGE_BUF_H_
#include <endian.h>
#include <cstring>
namespace android {
namespace nanohub {
/*
* Marshaling helper;
* deals with alignment and endianness.
* Assumption is:
* read*() parse buffer from device in LE format;
* return host endianness, aligned data
* write*() primitives take host endinnness, aligned data,
* generate buffer to be passed to device in LE format
*
* Primitives do minimal error checking, only to ensure buffer read/write
* safety. Caller is responsible for making sure correct amount of data
* has been processed.
*/
class MessageBuf {
char *data;
size_t size;
size_t pos;
bool readOnly;
public:
MessageBuf(char *buf, size_t bufSize) {
size = bufSize;
pos = 0;
data = buf;
readOnly = false;
}
MessageBuf(const char *buf, size_t bufSize) {
size = bufSize;
pos = 0;
data = const_cast<char *>(buf);
readOnly = true;
}
void reset() { pos = 0; }
const char *getData() const { return data; }
size_t getSize() const { return size; }
size_t getPos() const { return pos; }
size_t getRoom() const { return size - pos; }
uint8_t readU8() {
if (pos == size) {
return 0;
}
return data[pos++];
}
void writeU8(uint8_t val) {
if (pos == size || readOnly)
return;
data[pos++] = val;
}
uint16_t readU16() {
if (pos > (size - sizeof(uint16_t))) {
return 0;
}
uint16_t val;
memcpy(&val, &data[pos], sizeof(val));
pos += sizeof(val);
return le16toh(val);
}
void writeU16(uint16_t val) {
if (pos > (size - sizeof(uint16_t)) || readOnly) {
return;
}
uint16_t tmp = htole16(val);
memcpy(&data[pos], &tmp, sizeof(tmp));
pos += sizeof(tmp);
}
uint32_t readU32() {
if (pos > (size - sizeof(uint32_t))) {
return 0;
}
uint32_t val;
memcpy(&val, &data[pos], sizeof(val));
pos += sizeof(val);
return le32toh(val);
}
void writeU32(uint32_t val) {
if (pos > (size - sizeof(uint32_t)) || readOnly) {
return;
}
uint32_t tmp = htole32(val);
memcpy(&data[pos], &tmp, sizeof(tmp));
pos += sizeof(tmp);
}
uint64_t readU64() {
if (pos > (size - sizeof(uint64_t))) {
return 0;
}
uint64_t val;
memcpy(&val, &data[pos], sizeof(val));
pos += sizeof(val);
return le32toh(val);
}
void writeU64(uint64_t val) {
if (pos > (size - sizeof(uint64_t)) || readOnly) {
return;
}
uint64_t tmp = htole64(val);
memcpy(&data[pos], &tmp, sizeof(tmp));
pos += sizeof(tmp);
}
const void *readRaw(size_t bufSize) {
if (pos > (size - bufSize)) {
return nullptr;
}
const void *buf = &data[pos];
pos += bufSize;
return buf;
}
void writeRaw(const void *buf, size_t bufSize) {
if (pos > (size - bufSize) || readOnly) {
return;
}
memcpy(&data[pos], buf, bufSize);
pos += bufSize;
}
};
}; // namespace nanohub
}; // namespace android
#endif