summaryrefslogtreecommitdiffstats
path: root/src/core/hle/kernel/k_server_session.h
blob: 2876c231b2a1d58f8706800e3db985605635af82 (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
82
83
84
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later

#pragma once

#include <list>
#include <memory>
#include <string>
#include <utility>

#include "common/intrusive_list.h"

#include "core/hle/kernel/k_light_lock.h"
#include "core/hle/kernel/k_session_request.h"
#include "core/hle/kernel/k_synchronization_object.h"
#include "core/hle/result.h"

namespace Service {
class HLERequestContext;
class SessionRequestManager;
} // namespace Service

namespace Kernel {

class KernelCore;
class KSession;
class KThread;

class KServerSession final : public KSynchronizationObject,
                             public Common::IntrusiveListBaseNode<KServerSession> {
    KERNEL_AUTOOBJECT_TRAITS(KServerSession, KSynchronizationObject);

    friend class ServiceThread;

public:
    explicit KServerSession(KernelCore& kernel);
    ~KServerSession() override;

    void Destroy() override;

    void Initialize(KSession* p) {
        m_parent = p;
    }

    const KSession* GetParent() const {
        return m_parent;
    }

    bool IsSignaled() const override;
    void OnClientClosed();

    Result OnRequest(KSessionRequest* request);
    Result SendReply(uintptr_t server_message, uintptr_t server_buffer_size,
                     KPhysicalAddress server_message_paddr, bool is_hle = false);
    Result ReceiveRequest(uintptr_t server_message, uintptr_t server_buffer_size,
                          KPhysicalAddress server_message_paddr,
                          std::shared_ptr<Service::HLERequestContext>* out_context = nullptr,
                          std::weak_ptr<Service::SessionRequestManager> manager = {});

    Result SendReplyHLE() {
        R_RETURN(this->SendReply(0, 0, 0, true));
    }

    Result ReceiveRequestHLE(std::shared_ptr<Service::HLERequestContext>* out_context,
                             std::weak_ptr<Service::SessionRequestManager> manager) {
        R_RETURN(this->ReceiveRequest(0, 0, 0, out_context, manager));
    }

private:
    /// Frees up waiting client sessions when this server session is about to die
    void CleanupRequests();

    /// KSession that owns this KServerSession
    KSession* m_parent{};

    /// List of threads which are pending a reply.
    using RequestList = Common::IntrusiveListBaseTraits<KSessionRequest>::ListType;
    RequestList m_request_list{};
    KSessionRequest* m_current_request{};

    KLightLock m_lock;
};

} // namespace Kernel