You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
113 lines
3.2 KiB
113 lines
3.2 KiB
7 months ago
|
/*
|
||
|
* 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 <assert.h>
|
||
|
#include <inttypes.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <stdio.h>
|
||
|
|
||
|
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;
|
||
|
};
|