#include "stdafx.h" #include #include "Emu/NP/np_allocator.h" #include "Emu/NP/np_cache.h" namespace np { memberbin_cache::memberbin_cache(SceNpMatching2RoomMemberBinAttrInternal* sce_memberbin) { id = sce_memberbin->data.id; updateDate.tick = sce_memberbin->updateDate.tick; data.resize(sce_memberbin->data.size); memcpy(data.data(), sce_memberbin->data.ptr.get_ptr(), sce_memberbin->data.size); } member_cache::member_cache(const SceNpMatching2RoomMemberDataInternal* sce_member) { userInfo.npId = sce_member->userInfo.npId; if (sce_member->userInfo.onlineName) { userInfo.onlineName = *sce_member->userInfo.onlineName; } if (sce_member->userInfo.avatarUrl) { userInfo.avatarUrl = *sce_member->userInfo.avatarUrl; } joinDate.tick = sce_member->joinDate.tick; memberId = sce_member->memberId; teamId = sce_member->teamId; if (sce_member->roomGroup) { group_id = sce_member->roomGroup->groupId; } else { group_id = 0; } natType = sce_member->natType; flagAttr = sce_member->flagAttr; for (u32 i = 0; i < sce_member->roomMemberBinAttrInternalNum; i++) { bins.insert_or_assign(sce_member->roomMemberBinAttrInternal[i].data.id, memberbin_cache(&sce_member->roomMemberBinAttrInternal[i])); } } void room_cache::update(const SceNpMatching2RoomDataInternal* sce_roomdata) { num_slots = sce_roomdata->maxSlot; mask_password = sce_roomdata->passwordSlotMask; groups.clear(); for (u32 i = 0; i < sce_roomdata->roomGroupNum; i++) { const SceNpMatching2RoomGroup* sce_group = &sce_roomdata->roomGroup[i]; memcpy(&groups[sce_group->groupId], sce_group, sizeof(SceNpMatching2RoomGroup)); } members.clear(); vm::bptr sce_member = sce_roomdata->memberList.members; while (sce_member) { members.insert_or_assign(sce_member->memberId, member_cache(sce_member.get_ptr())); sce_member = sce_member->next; } if (sce_roomdata->memberList.me == sce_roomdata->memberList.owner) { owner = true; } } void cache_manager::insert_room(const SceNpMatching2RoomDataInternal* sce_roomdata) { std::lock_guard lock(mutex); rooms[sce_roomdata->roomId].update(sce_roomdata); } void cache_manager::add_member(SceNpMatching2RoomId room_id, const SceNpMatching2RoomMemberDataInternal* sce_roommemberdata) { std::lock_guard lock(mutex); ensure(rooms.contains(room_id), "cache_manager::add_member: Room not cached!"); rooms[room_id].members.insert_or_assign(sce_roommemberdata->memberId, member_cache(sce_roommemberdata)); } void cache_manager::del_member(SceNpMatching2RoomId room_id, SceNpMatching2RoomMemberId member_id) { std::lock_guard lock(mutex); ensure(rooms.contains(room_id), "cache_manager::del_member: Room not cached!"); rooms.erase(member_id); } void cache_manager::update_password(SceNpMatching2RoomId room_id, const std::optional& password) { std::lock_guard lock(mutex); rooms[room_id].password = password; } std::pair> cache_manager::get_slots(SceNpMatching2RoomId room_id) { std::lock_guard lock(mutex); if (!rooms.contains(room_id)) { return {SCE_NP_MATCHING2_ERROR_ROOM_NOT_FOUND, {}}; } const auto& room = rooms[room_id]; SceNpMatching2RoomSlotInfo slots; slots.roomId = room_id; SceNpMatching2RoomJoinedSlotMask join_mask = 0; for (const auto& member : room.members) { join_mask |= (1 << (member.first - 1)); } slots.joinedSlotMask = join_mask; slots.passwordSlotMask = room.mask_password; u64 joinable_slot_mask = (static_cast(1) << room.num_slots) - 1; u16 num_private_slots = std::popcount(room.mask_password & joinable_slot_mask); u16 num_public_slots = room.num_slots - num_private_slots; slots.publicSlotNum = num_public_slots; slots.privateSlotNum = num_private_slots; u16 open_private_slots = num_private_slots - std::popcount(join_mask & room.mask_password); u16 open_public_slots = num_public_slots - std::popcount(join_mask & (~room.mask_password)); slots.openPublicSlotNum = open_public_slots; slots.openPrivateSlotNum = open_private_slots; return {CELL_OK, slots}; } std::pair> cache_manager::get_memberids(u64 room_id, s32 sort_method) { std::lock_guard lock(mutex); if (!rooms.contains(room_id)) { return {SCE_NP_MATCHING2_ERROR_ROOM_NOT_FOUND, {}}; } const auto& room = rooms[room_id]; std::vector vec_memberids; switch (sort_method) { case SCE_NP_MATCHING2_SORT_METHOD_SLOT_NUMBER: { for (const auto& member : room.members) { vec_memberids.push_back(member.first); } break; } case SCE_NP_MATCHING2_SORT_METHOD_JOIN_DATE: { std::map map_joindate_id; for (const auto& member : room.members) { map_joindate_id.insert(std::make_pair(member.second.joinDate.tick, member.first)); } for (const auto& member : map_joindate_id) { vec_memberids.push_back(member.second); } break; } default: fmt::throw_exception("Unreachable"); } return {CELL_OK, vec_memberids}; } std::pair> cache_manager::get_password(SceNpMatching2RoomId room_id) { std::lock_guard lock(mutex); if (!rooms.contains(room_id)) { return {SCE_NP_MATCHING2_ERROR_ROOM_NOT_FOUND, {}}; } if (!rooms[room_id].owner) { return {SCE_NP_MATCHING2_ERROR_NOT_ALLOWED, {}}; } return {CELL_OK, rooms[room_id].password}; } error_code cache_manager::get_member_and_attrs(SceNpMatching2RoomId room_id, SceNpMatching2RoomMemberId member_id, const std::vector& binattrs_list, SceNpMatching2RoomMemberDataInternal* ptr_member, u32 addr_data, u32 size_data) { std::lock_guard lock(mutex); if (!rooms.contains(room_id)) { return SCE_NP_MATCHING2_ERROR_ROOM_NOT_FOUND; } if (!rooms[room_id].members.contains(member_id)) { return SCE_NP_MATCHING2_ERROR_ROOM_MEMBER_NOT_FOUND; } const auto& room = rooms.at(room_id); const auto& member = room.members.at(member_id); if (ptr_member) { memset(ptr_member, 0, sizeof(SceNpMatching2RoomMemberDataInternal)); memcpy(&ptr_member->userInfo.npId, &member.userInfo.npId, sizeof(SceNpId)); ptr_member->joinDate.tick = member.joinDate.tick; ptr_member->memberId = member.memberId; ptr_member->teamId = member.teamId; ptr_member->natType = member.natType; ptr_member->flagAttr = member.flagAttr; } const u32 needed_data_size = sizeof(SceNpOnlineName) + sizeof(SceNpAvatarUrl) + sizeof(SceNpMatching2RoomGroup) + (binattrs_list.size() * (sizeof(SceNpMatching2RoomMemberBinAttrInternal) + SCE_NP_MATCHING2_ROOMMEMBER_BIN_ATTR_INTERNAL_MAX_SIZE)); if (!addr_data || !ptr_member) { return needed_data_size; } if (size_data < needed_data_size) { return SCE_NP_MATCHING2_ERROR_INSUFFICIENT_BUFFER; } memory_allocator mem; mem.setup(vm::ptr(vm::cast(addr_data)), size_data); if (member.userInfo.onlineName) { ptr_member->userInfo.onlineName.set(mem.allocate(sizeof(SceNpOnlineName))); memcpy(ptr_member->userInfo.onlineName.get_ptr(), &member.userInfo.onlineName.value(), sizeof(SceNpOnlineName)); } if (member.userInfo.avatarUrl) { ptr_member->userInfo.avatarUrl.set(mem.allocate(sizeof(SceNpAvatarUrl))); memcpy(ptr_member->userInfo.avatarUrl.get_ptr(), &member.userInfo.avatarUrl.value(), sizeof(SceNpAvatarUrl)); } if (member.group_id) { ptr_member->roomGroup.set(mem.allocate(sizeof(SceNpMatching2RoomGroup))); memcpy(ptr_member->roomGroup.get_ptr(), &room.groups.at(member.group_id), sizeof(SceNpMatching2RoomGroup)); } u32 num_binattrs = 0; for (u32 i = 0; i < binattrs_list.size(); i++) { if (member.bins.contains(binattrs_list[i])) { num_binattrs++; } } if (num_binattrs) { ptr_member->roomMemberBinAttrInternal.set(mem.allocate(sizeof(SceNpMatching2RoomMemberBinAttrInternal) * num_binattrs)); ptr_member->roomMemberBinAttrInternalNum = num_binattrs; SceNpMatching2RoomMemberBinAttrInternal* bin_ptr = ptr_member->roomMemberBinAttrInternal.get_ptr(); u32 actual_cnt = 0; for (u32 i = 0; i < binattrs_list.size(); i++) { if (member.bins.contains(binattrs_list[i])) { const auto& bin = member.bins.at(binattrs_list[i]); bin_ptr[actual_cnt].updateDate.tick = bin.updateDate.tick; bin_ptr[actual_cnt].data.id = bin.id; bin_ptr[actual_cnt].data.size = bin.data.size(); bin_ptr[actual_cnt].data.ptr.set(mem.allocate(bin.data.size())); memcpy(bin_ptr[actual_cnt].data.ptr.get_ptr(), bin.data.data(), bin.data.size()); actual_cnt++; } } } return needed_data_size; } } // namespace np