#include "yellowstone_server.h" #include #include #include #include #include #include #include #include "hw/gpt.h" #include "hw/pcie.h" namespace { struct PartitionInfo { uint64_t device_id; uint64_t partition_lba; }; glcr::ErrorOr HandleDenaliRegistration(z_cap_t endpoint_cap) { GptReader reader( glcr::UniquePtr(new DenaliClient(endpoint_cap))); RET_ERR(reader.ParsePartitionTables()); return PartitionInfo{.device_id = 0, .partition_lba = reader.GetPrimaryPartitionLba()}; } } // namespace glcr::ErrorOr> YellowstoneServer::Create() { z_cap_t endpoint_cap; RET_ERR(ZEndpointCreate(&endpoint_cap)); ASSIGN_OR_RETURN(Mutex denali_mut, Mutex::Create()); RET_ERR(denali_mut.Lock()); ASSIGN_OR_RETURN(Mutex victoriafalls_mut, Mutex::Create()); RET_ERR(victoriafalls_mut.Lock()); return glcr::UniquePtr(new YellowstoneServer( endpoint_cap, glcr::Move(denali_mut), glcr::Move(victoriafalls_mut))); } YellowstoneServer::YellowstoneServer(z_cap_t endpoint_cap, Mutex&& denali_mutex, Mutex&& victoriafalls_mutex) : YellowstoneServerBase(endpoint_cap), has_denali_mutex_(glcr::Move(denali_mutex)), has_victoriafalls_mutex_(glcr::Move(victoriafalls_mutex)) {} glcr::ErrorCode YellowstoneServer::HandleGetAhciInfo(const Empty&, AhciInfo& info) { info.set_ahci_region(pci_reader_.GetAhciVmmo()); info.set_region_length(kPcieConfigurationSize); return glcr::OK; } glcr::ErrorCode YellowstoneServer::HandleGetFramebufferInfo( const Empty&, FramebufferInfo& info) { // FIXME: Don't do this for each request. OwnedMemoryRegion region = OwnedMemoryRegion::FromCapability(gBootFramebufferVmmoCap); ZFramebufferInfo* fb = reinterpret_cast(region.vaddr()); info.set_address_phys(fb->address_phys); info.set_width(fb->width); info.set_height(fb->height); info.set_pitch(fb->pitch); info.set_bpp(fb->bpp); info.set_memory_model(fb->memory_model); info.set_red_mask_size(fb->red_mask_size); info.set_red_mask_shift(fb->red_mask_shift); info.set_green_mask_size(fb->green_mask_size); info.set_green_mask_shift(fb->green_mask_shift); info.set_blue_mask_size(fb->blue_mask_size); info.set_blue_mask_shift(fb->blue_mask_shift); return glcr::OK; } glcr::ErrorCode YellowstoneServer::HandleGetDenali(const Empty&, DenaliInfo& info) { if (!endpoint_map_.Contains("denali")) { return glcr::NOT_FOUND; } z_cap_t new_denali; check(ZCapDuplicate(endpoint_map_.at("denali"), kZionPerm_All, &new_denali)); info.set_denali_endpoint(new_denali); info.set_device_id(device_id_); info.set_lba_offset(lba_offset_); return glcr::OK; } glcr::ErrorCode YellowstoneServer::HandleRegisterEndpoint( const RegisterEndpointRequest& req, Empty&) { dbgln("Registering {}.", req.endpoint_name().view()); check(endpoint_map_.Insert(req.endpoint_name(), req.endpoint_capability())); if (req.endpoint_name() == "denali") { auto part_info_or = HandleDenaliRegistration(req.endpoint_capability()); if (!part_info_or.ok()) { check(part_info_or.error()); } device_id_ = part_info_or.value().device_id; lba_offset_ = part_info_or.value().partition_lba; check(has_denali_mutex_.Release()); } else if (req.endpoint_name() == "victoriafalls") { // FIXME: Probably make a separate copy for use within yellowstone vs // transmit to other processes. vfs_client_ = glcr::MakeShared(req.endpoint_capability()); check(has_victoriafalls_mutex_.Release()); } else { dbgln("[WARN] Got endpoint cap type: {}", req.endpoint_name().cstr()); } return glcr::OK; } glcr::ErrorCode YellowstoneServer::HandleGetEndpoint( const GetEndpointRequest& req, Endpoint& resp) { if (!endpoint_map_.Contains(req.endpoint_name())) { return glcr::NOT_FOUND; } z_cap_t cap = endpoint_map_.at(req.endpoint_name()); z_cap_t new_cap; check(ZCapDuplicate(cap, kZionPerm_All, &new_cap)); resp.set_endpoint(new_cap); return glcr::OK; } glcr::ErrorCode YellowstoneServer::WaitDenaliRegistered() { RET_ERR(has_denali_mutex_.Lock()); return has_denali_mutex_.Release(); } glcr::ErrorCode YellowstoneServer::WaitVictoriaFallsRegistered() { RET_ERR(has_victoriafalls_mutex_.Lock()); return has_victoriafalls_mutex_.Release(); } glcr::SharedPtr YellowstoneServer::GetVFSClient() { return vfs_client_; }