2023-06-15 16:20:29 -07:00
|
|
|
#include "denali_server.h"
|
|
|
|
|
|
|
|
#include <mammoth/debug.h>
|
|
|
|
#include <zcall.h>
|
|
|
|
|
2023-06-16 01:31:23 -07:00
|
|
|
namespace {
|
|
|
|
DenaliServer* gServer = nullptr;
|
|
|
|
void HandleResponse(uint64_t lba, uint64_t size, uint64_t cap) {
|
|
|
|
gServer->HandleResponse(lba, size, cap);
|
|
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
2023-06-15 16:20:29 -07:00
|
|
|
DenaliServer::DenaliServer(uint64_t channel_cap, AhciDriver& driver)
|
2023-06-16 01:31:23 -07:00
|
|
|
: channel_cap_(channel_cap), driver_(driver) {
|
|
|
|
gServer = this;
|
|
|
|
}
|
2023-06-15 16:20:29 -07:00
|
|
|
|
|
|
|
z_err_t DenaliServer::RunServer() {
|
|
|
|
while (true) {
|
|
|
|
uint64_t buff_size = kBuffSize;
|
|
|
|
uint64_t cap_size = 0;
|
2023-06-20 14:41:44 -07:00
|
|
|
RET_ERR(ZChannelRecv(channel_cap_, &buff_size, read_buffer_, &cap_size,
|
|
|
|
nullptr));
|
2023-06-17 02:01:21 -07:00
|
|
|
if (buff_size < sizeof(uint64_t)) {
|
|
|
|
dbgln("Skipping invalid message");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
uint64_t type = *reinterpret_cast<uint64_t*>(read_buffer_);
|
2023-06-15 16:20:29 -07:00
|
|
|
switch (type) {
|
|
|
|
case Z_INVALID:
|
|
|
|
dbgln(reinterpret_cast<char*>(read_buffer_));
|
|
|
|
break;
|
|
|
|
case DENALI_READ: {
|
|
|
|
DenaliRead* read_req = reinterpret_cast<DenaliRead*>(read_buffer_);
|
|
|
|
uint64_t memcap = 0;
|
2023-06-16 01:31:23 -07:00
|
|
|
RET_ERR(HandleRead(*read_req));
|
2023-06-15 16:20:29 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
dbgln("Invalid message type.");
|
|
|
|
return Z_ERR_UNIMPLEMENTED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-16 01:31:23 -07:00
|
|
|
z_err_t DenaliServer::HandleRead(const DenaliRead& read) {
|
|
|
|
AhciDevice* device;
|
|
|
|
RET_ERR(driver_.GetDevice(read.device_id, &device));
|
|
|
|
|
|
|
|
device->IssueCommand(
|
|
|
|
new DmaReadCommand(read.lba, read.size, ::HandleResponse));
|
2023-06-15 16:20:29 -07:00
|
|
|
|
|
|
|
return Z_OK;
|
|
|
|
}
|
2023-06-16 01:31:23 -07:00
|
|
|
|
|
|
|
void DenaliServer::HandleResponse(uint64_t lba, uint64_t size, uint64_t cap) {
|
|
|
|
DenaliReadResponse resp{
|
|
|
|
.device_id = 0,
|
|
|
|
.lba = lba,
|
|
|
|
.size = size,
|
|
|
|
};
|
2023-06-17 02:01:21 -07:00
|
|
|
check(ZChannelSend(channel_cap_, sizeof(resp),
|
2023-06-16 01:31:23 -07:00
|
|
|
reinterpret_cast<uint8_t*>(&resp), 1, &cap));
|
|
|
|
}
|