mirror of
https://github.com/ndeadly/MissionControl
synced 2025-01-09 03:08:41 +00:00
223 lines
6.4 KiB
C++
223 lines
6.4 KiB
C++
#include <mutex>
|
|
#include <cstring>
|
|
#include "bluetooth_circularbuffer.hpp"
|
|
|
|
#include "../btdrv_mitm_logging.hpp"
|
|
|
|
namespace ams::bluetooth {
|
|
|
|
CircularBuffer::CircularBuffer(void) {
|
|
this->readOffset = 0;
|
|
this->writeOffset = 0;
|
|
this->isInitialized = false;
|
|
this->event = nullptr;
|
|
}
|
|
|
|
void CircularBuffer::Initialize(const char *name) {
|
|
if (!name || this->isInitialized)
|
|
fatalThrow(-1);
|
|
|
|
this->readOffset = 0;
|
|
this->writeOffset = 0;
|
|
std::strncpy(this->name, name, sizeof(this->name) - 1);
|
|
this->_unk1 = 0;
|
|
this->size = BLUETOOTH_CIRCBUFFER_SIZE;
|
|
this->isInitialized = true;
|
|
}
|
|
|
|
void CircularBuffer::Finalize(void) {
|
|
if (!this->isInitialized)
|
|
fatalThrow(-1);
|
|
|
|
this->isInitialized = false;
|
|
this->event = nullptr;
|
|
}
|
|
|
|
bool CircularBuffer::IsInitialized(void) {
|
|
return this->isInitialized;
|
|
}
|
|
|
|
u64 CircularBuffer::GetWriteableSize(void) {
|
|
u32 readOffset = this->readOffset;
|
|
u32 writeOffset = this->writeOffset;
|
|
|
|
if (!this->isInitialized)
|
|
return 0;
|
|
|
|
u64 size;
|
|
if (readOffset <= writeOffset)
|
|
size = (BLUETOOTH_CIRCBUFFER_SIZE - 1) - writeOffset + readOffset;
|
|
else
|
|
size = readOffset - writeOffset - 1;
|
|
|
|
if (size > BLUETOOTH_CIRCBUFFER_SIZE)
|
|
size = 0;
|
|
|
|
return size;
|
|
}
|
|
|
|
void CircularBuffer::SetWriteCompleteEvent(os::EventType *event) {
|
|
this->event = event;
|
|
}
|
|
|
|
u64 CircularBuffer::Write(u8 type, void *data, size_t size) {
|
|
|
|
if (!this->isInitialized)
|
|
return -1;
|
|
|
|
std::scoped_lock lk(this->mutex);
|
|
|
|
ON_SCOPE_EXIT {
|
|
if (this->event)
|
|
os::SignalEvent(this->event);
|
|
};
|
|
|
|
if (size + sizeof(CircularBufferPacketHeader) <= this->GetWriteableSize()) {
|
|
if (size + 2*sizeof(CircularBufferPacketHeader) > BLUETOOTH_CIRCBUFFER_SIZE - this->writeOffset) {
|
|
R_TRY(this->_write(0xff, nullptr, (BLUETOOTH_CIRCBUFFER_SIZE - this->writeOffset) - sizeof(CircularBufferPacketHeader)));
|
|
}
|
|
|
|
R_TRY(this->_write(type, data, size));
|
|
this->_updateUtilization();
|
|
|
|
return 0;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
void CircularBuffer::DiscardOldPackets(u8 type, u32 ageLimit) {
|
|
if (this->isInitialized) {
|
|
|
|
CircularBufferPacket *packet;
|
|
TimeSpan timespan;
|
|
do {
|
|
if (this->readOffset == this->writeOffset)
|
|
return;
|
|
|
|
packet = reinterpret_cast<CircularBufferPacket *>(&this->data[this->readOffset]);
|
|
if (packet->header.type != 0xff) {
|
|
|
|
if (packet->header.type != type)
|
|
return;
|
|
|
|
timespan = os::ConvertToTimeSpan(os::GetSystemTick() - packet->header.timestamp);
|
|
if (timespan.GetMilliSeconds() <= ageLimit)
|
|
return;
|
|
}
|
|
|
|
this->Free();
|
|
} while (this->isInitialized);
|
|
}
|
|
}
|
|
|
|
CircularBufferPacket *CircularBuffer::Read(void) {
|
|
return this->_read();
|
|
}
|
|
|
|
u64 CircularBuffer::Free(void) {
|
|
if (!this->isInitialized)
|
|
return -1;
|
|
|
|
if (this->readOffset == this->writeOffset)
|
|
return -1;
|
|
|
|
auto packet = reinterpret_cast<CircularBufferPacket *>(&this->data[this->readOffset]);
|
|
u32 newOffset = this->readOffset + packet->header.size + sizeof(packet->header);
|
|
|
|
if (newOffset >= BLUETOOTH_CIRCBUFFER_SIZE)
|
|
newOffset = 0;
|
|
|
|
if (newOffset < BLUETOOTH_CIRCBUFFER_SIZE) {
|
|
this->readOffset = newOffset;
|
|
return 0;
|
|
}
|
|
|
|
fatalThrow(-1);
|
|
}
|
|
|
|
void CircularBuffer::_setReadOffset(u32 offset) {
|
|
if (offset >= BLUETOOTH_CIRCBUFFER_SIZE)
|
|
fatalThrow(-1);
|
|
|
|
this->readOffset = offset;
|
|
}
|
|
|
|
void CircularBuffer::_setWriteOffset(u32 offset) {
|
|
if (offset >= BLUETOOTH_CIRCBUFFER_SIZE)
|
|
fatalThrow(-1);
|
|
|
|
this->writeOffset = offset;
|
|
}
|
|
|
|
u32 CircularBuffer::_getWriteOffset(void) {
|
|
return this->writeOffset;
|
|
}
|
|
|
|
u32 CircularBuffer::_getReadOffset(void) {
|
|
return this->readOffset;
|
|
}
|
|
|
|
u64 CircularBuffer::_write(u8 type, void *data, size_t size) {
|
|
auto packet = reinterpret_cast<CircularBufferPacket *>(&this->data[this->writeOffset]);
|
|
packet->header.type = type;
|
|
packet->header.timestamp = os::GetSystemTick();
|
|
packet->header.size = size;
|
|
|
|
if (type != 0xff) {
|
|
if (data && (size > 0))
|
|
memcpy(&packet->data, data, size);
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
u32 newOffset = this->writeOffset + size + sizeof(CircularBufferPacketHeader);
|
|
if (newOffset > BLUETOOTH_CIRCBUFFER_SIZE)
|
|
return -1;
|
|
|
|
if (newOffset == BLUETOOTH_CIRCBUFFER_SIZE)
|
|
this->writeOffset = 0;
|
|
else
|
|
this->writeOffset = newOffset;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void CircularBuffer::_updateUtilization(void) {
|
|
u32 newCapacity = this->isInitialized ? this->GetWriteableSize() : 0;
|
|
|
|
if (this->size > newCapacity + 1000)
|
|
this->size = newCapacity;
|
|
}
|
|
|
|
CircularBufferPacket *CircularBuffer::_read(void) {
|
|
if (this->isInitialized) {
|
|
CircularBufferPacket *packet;
|
|
u32 newOffset;
|
|
do {
|
|
if (this->readOffset == this->writeOffset)
|
|
return nullptr;
|
|
|
|
packet = reinterpret_cast<CircularBufferPacket *>(&this->data[this->readOffset]);
|
|
|
|
if (packet->header.type != 0xff)
|
|
return packet;
|
|
|
|
if (!this->isInitialized)
|
|
return nullptr;
|
|
|
|
if (this->readOffset != this->writeOffset) {
|
|
newOffset = this->readOffset + packet->header.size + sizeof(packet->header);
|
|
if (newOffset >= BLUETOOTH_CIRCBUFFER_SIZE)
|
|
newOffset = 0;
|
|
|
|
this->_setReadOffset(newOffset);
|
|
}
|
|
|
|
} while (this->isInitialized);
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
}
|