/* * Copyright (C) 2011 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. */ #pragma once #include "ErrorLog.h" #include "base/Stream.h" #include #include #include #include class IOStream { protected: explicit IOStream(size_t bufSize) : m_bufsize(bufSize) {} ~IOStream() { // NOTE: m_buf was owned by the child class thus we expect it to be // released before the object destruction. } public: virtual void *allocBuffer(size_t minSize) = 0; virtual int commitBuffer(size_t size) = 0; virtual int writeFully(const void* buf, size_t len) = 0; virtual const unsigned char *readFully( void *buf, size_t len) = 0; size_t read(void* buf, size_t bufLen) { if (!readRaw(buf, &bufLen)) { return 0; } return bufLen; } unsigned char* alloc(size_t len) { if (m_buf && len > m_free) { if (flush() < 0) { ERR("Failed to flush in alloc\n"); return NULL; // we failed to flush so something is wrong } } if (!m_buf || len > m_bufsize) { int allocLen = m_bufsize < len ? len : m_bufsize; m_buf = (unsigned char *)allocBuffer(allocLen); if (!m_buf) { ERR("Alloc (%u bytes) failed\n", allocLen); return NULL; } m_bufsize = m_free = allocLen; } unsigned char* ptr = m_buf + (m_bufsize - m_free); m_free -= len; return ptr; } int flush() { if (!m_buf || m_free == m_bufsize) return 0; int stat = commitBuffer(m_bufsize - m_free); m_buf = NULL; m_free = 0; return stat; } const unsigned char *readback(void *buf, size_t len) { flush(); return readFully(buf, len); } void save(android::base::Stream* stream) { stream->putBe32(m_bufsize); stream->putBe32(m_free); stream->putByte(m_buf != nullptr); onSave(stream); } void load(android::base::Stream* stream) { m_bufsize = stream->getBe32(); m_free = stream->getBe32(); const bool haveBuf = stream->getByte(); const auto buf = onLoad(stream); m_buf = haveBuf ? buf : nullptr; } virtual void* getDmaForReading(uint64_t guest_paddr) = 0; virtual void unlockDma(uint64_t guest_paddr) = 0; protected: virtual const unsigned char *readRaw(void *buf, size_t *inout_len) = 0; virtual void onSave(android::base::Stream* stream) = 0; virtual unsigned char* onLoad(android::base::Stream* stream) = 0; unsigned char* m_buf = nullptr; size_t m_bufsize; size_t m_free = 0; };