#include "denali_server.h" #include #include #include #include glcr::ErrorOr> DenaliServer::Create( AhciDriver& driver) { z_cap_t cap; RET_ERR(ZEndpointCreate(&cap)); return glcr::UniquePtr(new DenaliServer(cap, driver)); } glcr::Status DenaliServer::HandleRead(const ReadRequest& req, ReadResponse& resp) { ASSIGN_OR_RETURN(AhciDevice * device, driver_.GetDevice(req.device_id())); uint64_t paddr; mmth::OwnedMemoryRegion region = mmth::OwnedMemoryRegion::ContiguousPhysical(req.size() * 512, &paddr); DmaReadCommand command(req.lba(), req.size(), paddr); device->IssueCommand(&command); command.WaitComplete(); resp.set_device_id(req.device_id()); resp.set_size(req.size()); resp.set_memory(region.DuplicateCap()); return glcr::Status::Ok(); } glcr::Status DenaliServer::HandleReadMany(const ReadManyRequest& req, ReadResponse& resp) { ASSIGN_OR_RETURN(AhciDevice * device, driver_.GetDevice(req.device_id())); if (req.lba().size() != req.sector_cnt().size()) { return glcr::InvalidArgument("LBA and Sector Cnt must be the same length."); } uint64_t sector_cnt = 0; for (uint64_t i = 0; i < req.sector_cnt().size(); i++) { sector_cnt += req.sector_cnt().at(i); } uint64_t region_paddr; mmth::OwnedMemoryRegion region = mmth::OwnedMemoryRegion::ContiguousPhysical( sector_cnt * 512, ®ion_paddr); for (uint64_t i = 0; i < req.lba().size(); i++) { uint64_t lba = req.lba().at(i); uint64_t size = req.sector_cnt().at(i); DmaReadCommand command(lba, size, region_paddr); device->IssueCommand(&command); command.WaitComplete(); region_paddr += size * 512; } resp.set_device_id(req.device_id()); resp.set_size(sector_cnt); resp.set_memory(region.DuplicateCap()); return glcr::Status::Ok(); }