yuzu/src/core/hle/kernel/kernel.h

153 lines
3.5 KiB
C
Raw Normal View History

2014-05-09 22:11:18 -04:00
// Copyright 2014 Citra Emulator Project / PPSSPP Project
2014-12-16 21:38:14 -08:00
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.
2014-05-09 22:11:18 -04:00
#pragma once
2015-06-21 13:40:28 +01:00
#include <cstddef>
#include <string>
#include <utility>
#include <boost/smart_ptr/intrusive_ptr.hpp>
#include "common/assert.h"
2015-05-06 04:06:12 -03:00
#include "common/common_types.h"
2014-05-09 22:11:18 -04:00
namespace Kernel {
using Handle = u32;
enum class HandleType : u32 {
2016-11-30 23:28:31 -05:00
Unknown,
Event,
Mutex,
SharedMemory,
Thread,
Process,
AddressArbiter,
ConditionVariable,
2016-11-30 23:28:31 -05:00
Timer,
ResourceLimit,
CodeSet,
ClientPort,
ServerPort,
ClientSession,
ServerSession,
Domain,
};
enum {
DEFAULT_STACK_SIZE = 0x4000,
};
2016-05-25 13:36:15 +03:00
enum class ResetType {
OneShot,
Sticky,
Pulse,
};
class Object : NonCopyable {
2014-05-09 22:11:18 -04:00
public:
virtual ~Object() {}
/// Returns a unique identifier for the object. For debugging purposes only.
unsigned int GetObjectId() const {
return object_id;
}
virtual std::string GetTypeName() const {
return "[BAD KERNEL OBJECT TYPE]";
}
virtual std::string GetName() const {
return "[UNKNOWN KERNEL OBJECT]";
}
virtual Kernel::HandleType GetHandleType() const = 0;
/**
* Check if a thread can wait on the object
* @return True if a thread can wait on the object, otherwise false
*/
bool IsWaitable() const {
switch (GetHandleType()) {
case HandleType::Event:
case HandleType::Mutex:
case HandleType::Thread:
case HandleType::ConditionVariable:
case HandleType::Timer:
case HandleType::ServerPort:
case HandleType::ServerSession:
return true;
case HandleType::Unknown:
case HandleType::SharedMemory:
case HandleType::Process:
case HandleType::AddressArbiter:
case HandleType::ResourceLimit:
case HandleType::CodeSet:
case HandleType::ClientPort:
case HandleType::ClientSession:
case HandleType::Domain:
return false;
}
UNREACHABLE();
}
2014-05-09 22:11:18 -04:00
/**
* Check if svcSendSyncRequest can be called on the object
* @return True svcSendSyncRequest can be called on the object, otherwise false
*/
bool IsSyncable() const {
switch (GetHandleType()) {
case HandleType::ClientSession:
case HandleType::Domain:
return true;
}
UNREACHABLE();
}
public:
static unsigned int next_object_id;
private:
friend void intrusive_ptr_add_ref(Object*);
friend void intrusive_ptr_release(Object*);
2014-05-09 22:11:18 -04:00
unsigned int ref_count = 0;
unsigned int object_id = next_object_id++;
};
2014-05-09 22:11:18 -04:00
// Special functions used by boost::instrusive_ptr to do automatic ref-counting
inline void intrusive_ptr_add_ref(Object* object) {
++object->ref_count;
}
2014-05-09 22:11:18 -04:00
inline void intrusive_ptr_release(Object* object) {
if (--object->ref_count == 0) {
delete object;
2014-11-18 08:27:16 -05:00
}
}
2014-05-09 22:11:18 -04:00
template <typename T>
using SharedPtr = boost::intrusive_ptr<T>;
/**
* Attempts to downcast the given Object pointer to a pointer to T.
* @return Derived pointer to the object, or `nullptr` if `object` isn't of type T.
*/
template <typename T>
inline SharedPtr<T> DynamicObjectCast(SharedPtr<Object> object) {
if (object != nullptr && object->GetHandleType() == T::HANDLE_TYPE) {
return boost::static_pointer_cast<T>(std::move(object));
}
return nullptr;
}
/// Initialize the kernel with the specified system mode.
void Init(u32 system_mode);
/// Shutdown the kernel
void Shutdown();
} // namespace Kernel