summaryrefslogtreecommitdiffstats
path: root/src/core/hle/service/bcat/delivery_cache_directory_service.cpp
blob: 8bda1168a7ed2dc3f60c8581e23e6ac78a708736 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
// SPDX-License-Identifier: GPL-3.0-or-later

#include "common/string_util.h"
#include "core/file_sys/vfs/vfs_types.h"
#include "core/hle/service/bcat/bcat_result.h"
#include "core/hle/service/bcat/bcat_util.h"
#include "core/hle/service/bcat/delivery_cache_directory_service.h"
#include "core/hle/service/cmif_serialization.h"

namespace Service::BCAT {

// The digest is only used to determine if a file is unique compared to others of the same name.
// Since the algorithm isn't ever checked in game, MD5 is safe.
BcatDigest DigestFile(const FileSys::VirtualFile& file) {
    BcatDigest out{};
    const auto bytes = file->ReadAllBytes();
    mbedtls_md5_ret(bytes.data(), bytes.size(), out.data());
    return out;
}

IDeliveryCacheDirectoryService::IDeliveryCacheDirectoryService(Core::System& system_,
                                                               FileSys::VirtualDir root_)
    : ServiceFramework{system_, "IDeliveryCacheDirectoryService"}, root(std::move(root_)) {
    // clang-format off
    static const FunctionInfo functions[] = {
        {0, C<&IDeliveryCacheDirectoryService::Open>, "Open"},
        {1, C<&IDeliveryCacheDirectoryService::Read>, "Read"},
        {2, C<&IDeliveryCacheDirectoryService::GetCount>, "GetCount"},
    };
    // clang-format on

    RegisterHandlers(functions);
}

IDeliveryCacheDirectoryService::~IDeliveryCacheDirectoryService() = default;

Result IDeliveryCacheDirectoryService::Open(DirectoryName dir_name_raw) {
    const auto dir_name =
        Common::StringFromFixedZeroTerminatedBuffer(dir_name_raw.data(), dir_name_raw.size());

    LOG_DEBUG(Service_BCAT, "called, dir_name={}", dir_name);

    // R_TRY(VerifyNameValidDir(dir_name_raw));
    R_UNLESS(current_dir == nullptr, ResultEntityAlreadyOpen);

    const auto dir = root->GetSubdirectory(dir_name);
    R_UNLESS(dir != nullptr, ResultFailedOpenEntity);

    R_SUCCEED();
}

Result IDeliveryCacheDirectoryService::Read(
    Out<u32> out_buffer_size,
    OutArray<DeliveryCacheDirectoryEntry, BufferAttr_HipcMapAlias> out_buffer) {
    LOG_DEBUG(Service_BCAT, "called, write_size={:016X}", out_buffer.size());

    R_UNLESS(current_dir != nullptr, ResultNoOpenEntry);

    const auto files = current_dir->GetFiles();
    *out_buffer_size = static_cast<u32>(std::min(files.size(), out_buffer.size()));
    std::transform(files.begin(), files.begin() + *out_buffer_size, out_buffer.begin(),
                   [](const auto& file) {
                       FileName name{};
                       std::memcpy(name.data(), file->GetName().data(),
                                   std::min(file->GetName().size(), name.size()));
                       return DeliveryCacheDirectoryEntry{name, file->GetSize(), DigestFile(file)};
                   });
    R_SUCCEED();
}

Result IDeliveryCacheDirectoryService::GetCount(Out<u32> out_count) {
    LOG_DEBUG(Service_BCAT, "called");

    R_UNLESS(current_dir != nullptr, ResultNoOpenEntry);

    *out_count = static_cast<u32>(current_dir->GetFiles().size());
    R_SUCCEED();
}

} // namespace Service::BCAT