diff options
| author | bunnei <bunneidev@gmail.com> | 2014-12-28 21:11:55 -0500 | 
|---|---|---|
| committer | bunnei <bunneidev@gmail.com> | 2014-12-28 21:11:55 -0500 | 
| commit | 77363d9590ed544cf714f26c575e98b7858e18e1 (patch) | |
| tree | a48225a3b62bb3d72d94289b6e34326cbed4f5cc /src/core/hle/kernel | |
| parent | 4bf803579f52633e7e9f02c0fc99116f89331b8d (diff) | |
| parent | 7e2903cb74050d846f2da951dff7e84aee13761b (diff) | |
Merge pull request #331 from yuriks/handle-reform
New Handle manager
Diffstat (limited to 'src/core/hle/kernel')
| -rw-r--r-- | src/core/hle/kernel/address_arbiter.cpp | 7 | ||||
| -rw-r--r-- | src/core/hle/kernel/event.cpp | 15 | ||||
| -rw-r--r-- | src/core/hle/kernel/kernel.cpp | 124 | ||||
| -rw-r--r-- | src/core/hle/kernel/kernel.h | 196 | ||||
| -rw-r--r-- | src/core/hle/kernel/mutex.cpp | 11 | ||||
| -rw-r--r-- | src/core/hle/kernel/semaphore.cpp | 9 | ||||
| -rw-r--r-- | src/core/hle/kernel/session.h | 4 | ||||
| -rw-r--r-- | src/core/hle/kernel/shared_memory.cpp | 11 | ||||
| -rw-r--r-- | src/core/hle/kernel/thread.cpp | 27 | ||||
| -rw-r--r-- | src/core/hle/kernel/thread.h | 3 | 
10 files changed, 227 insertions, 180 deletions
| diff --git a/src/core/hle/kernel/address_arbiter.cpp b/src/core/hle/kernel/address_arbiter.cpp index 77491900a..38705e3cd 100644 --- a/src/core/hle/kernel/address_arbiter.cpp +++ b/src/core/hle/kernel/address_arbiter.cpp @@ -20,8 +20,8 @@ public:      std::string GetTypeName() const override { return "Arbiter"; }      std::string GetName() const override { return name; } -    static Kernel::HandleType GetStaticHandleType() { return HandleType::AddressArbiter; } -    Kernel::HandleType GetHandleType() const override { return HandleType::AddressArbiter; } +    static const HandleType HANDLE_TYPE = HandleType::AddressArbiter; +    HandleType GetHandleType() const override { return HANDLE_TYPE; }      std::string name;   ///< Name of address arbiter object (optional)  }; @@ -62,7 +62,8 @@ ResultCode ArbitrateAddress(Handle handle, ArbitrationType type, u32 address, s3  /// Create an address arbiter  AddressArbiter* CreateAddressArbiter(Handle& handle, const std::string& name) {      AddressArbiter* address_arbiter = new AddressArbiter; -    handle = Kernel::g_object_pool.Create(address_arbiter); +    // TOOD(yuriks): Fix error reporting +    handle = Kernel::g_handle_table.Create(address_arbiter).ValueOr(INVALID_HANDLE);      address_arbiter->name = name;      return address_arbiter;  } diff --git a/src/core/hle/kernel/event.cpp b/src/core/hle/kernel/event.cpp index 4de3fab3c..e43c3ee4e 100644 --- a/src/core/hle/kernel/event.cpp +++ b/src/core/hle/kernel/event.cpp @@ -19,8 +19,8 @@ public:      std::string GetTypeName() const override { return "Event"; }      std::string GetName() const override { return name; } -    static Kernel::HandleType GetStaticHandleType() { return Kernel::HandleType::Event; } -    Kernel::HandleType GetHandleType() const override { return Kernel::HandleType::Event; } +    static const HandleType HANDLE_TYPE = HandleType::Event; +    HandleType GetHandleType() const override { return HANDLE_TYPE; }      ResetType intitial_reset_type;          ///< ResetType specified at Event initialization      ResetType reset_type;                   ///< Current ResetType @@ -53,7 +53,7 @@ public:   * @return Result of operation, 0 on success, otherwise error code   */  ResultCode SetPermanentLock(Handle handle, const bool permanent_locked) { -    Event* evt = g_object_pool.Get<Event>(handle); +    Event* evt = g_handle_table.Get<Event>(handle);      if (evt == nullptr) return InvalidHandle(ErrorModule::Kernel);      evt->permanent_locked = permanent_locked; @@ -67,7 +67,7 @@ ResultCode SetPermanentLock(Handle handle, const bool permanent_locked) {   * @return Result of operation, 0 on success, otherwise error code   */  ResultCode SetEventLocked(const Handle handle, const bool locked) { -    Event* evt = g_object_pool.Get<Event>(handle); +    Event* evt = g_handle_table.Get<Event>(handle);      if (evt == nullptr) return InvalidHandle(ErrorModule::Kernel);      if (!evt->permanent_locked) { @@ -82,7 +82,7 @@ ResultCode SetEventLocked(const Handle handle, const bool locked) {   * @return Result of operation, 0 on success, otherwise error code   */  ResultCode SignalEvent(const Handle handle) { -    Event* evt = g_object_pool.Get<Event>(handle); +    Event* evt = g_handle_table.Get<Event>(handle);      if (evt == nullptr) return InvalidHandle(ErrorModule::Kernel);      // Resume threads waiting for event to signal @@ -110,7 +110,7 @@ ResultCode SignalEvent(const Handle handle) {   * @return Result of operation, 0 on success, otherwise error code   */  ResultCode ClearEvent(Handle handle) { -    Event* evt = g_object_pool.Get<Event>(handle); +    Event* evt = g_handle_table.Get<Event>(handle);      if (evt == nullptr) return InvalidHandle(ErrorModule::Kernel);      if (!evt->permanent_locked) { @@ -129,7 +129,8 @@ ResultCode ClearEvent(Handle handle) {  Event* CreateEvent(Handle& handle, const ResetType reset_type, const std::string& name) {      Event* evt = new Event; -    handle = Kernel::g_object_pool.Create(evt); +    // TOOD(yuriks): Fix error reporting +    handle = Kernel::g_handle_table.Create(evt).ValueOr(INVALID_HANDLE);      evt->locked = true;      evt->permanent_locked = false; diff --git a/src/core/hle/kernel/kernel.cpp b/src/core/hle/kernel/kernel.cpp index 5fd06046e..e59ed1b57 100644 --- a/src/core/hle/kernel/kernel.cpp +++ b/src/core/hle/kernel/kernel.cpp @@ -13,77 +13,93 @@  namespace Kernel {  Handle g_main_thread = 0; -ObjectPool g_object_pool; +HandleTable g_handle_table;  u64 g_program_id = 0; -ObjectPool::ObjectPool() { -    next_id = INITIAL_NEXT_ID; +HandleTable::HandleTable() { +    next_generation = 1; +    Clear();  } -Handle ObjectPool::Create(Object* obj, int range_bottom, int range_top) { -    if (range_top > MAX_COUNT) { -        range_top = MAX_COUNT; -    } -    if (next_id >= range_bottom && next_id < range_top) { -        range_bottom = next_id++; -    } -    for (int i = range_bottom; i < range_top; i++) { -        if (!occupied[i]) { -            occupied[i] = true; -            pool[i] = obj; -            pool[i]->handle = i + HANDLE_OFFSET; -            return i + HANDLE_OFFSET; -        } +ResultVal<Handle> HandleTable::Create(Object* obj) { +    _dbg_assert_(Kernel, obj != nullptr); + +    u16 slot = next_free_slot; +    if (slot >= generations.size()) { +        LOG_ERROR(Kernel, "Unable to allocate Handle, too many slots in use."); +        return ERR_OUT_OF_HANDLES;      } -    LOG_ERROR(Kernel, "Unable to allocate kernel object, too many objects slots in use."); -    return 0; -} +    next_free_slot = generations[slot]; -bool ObjectPool::IsValid(Handle handle) const { -    int index = handle - HANDLE_OFFSET; -    if (index < 0) -        return false; -    if (index >= MAX_COUNT) -        return false; +    u16 generation = next_generation++; -    return occupied[index]; +    // Overflow count so it fits in the 15 bits dedicated to the generation in the handle. +    // CTR-OS doesn't use generation 0, so skip straight to 1. +    if (next_generation >= (1 << 15)) next_generation = 1; + +    generations[slot] = generation; +    intrusive_ptr_add_ref(obj); +    objects[slot] = obj; + +    Handle handle = generation | (slot << 15); +    obj->handle = handle; +    return MakeResult<Handle>(handle);  } -void ObjectPool::Clear() { -    for (int i = 0; i < MAX_COUNT; i++) { -        //brutally clear everything, no validation -        if (occupied[i]) -            delete pool[i]; -        occupied[i] = false; +ResultVal<Handle> HandleTable::Duplicate(Handle handle) { +    Object* object = GetGeneric(handle); +    if (object == nullptr) { +        LOG_ERROR(Kernel, "Tried to duplicate invalid handle: %08X", handle); +        return ERR_INVALID_HANDLE;      } -    pool.fill(nullptr); -    next_id = INITIAL_NEXT_ID; +    return Create(object);  } -Object* &ObjectPool::operator [](Handle handle) -{ -    _dbg_assert_msg_(Kernel, IsValid(handle), "GRABBING UNALLOCED KERNEL OBJ"); -    return pool[handle - HANDLE_OFFSET]; +ResultCode HandleTable::Close(Handle handle) { +    if (!IsValid(handle)) +        return ERR_INVALID_HANDLE; + +    size_t slot = GetSlot(handle); +    u16 generation = GetGeneration(handle); + +    intrusive_ptr_release(objects[slot]); +    objects[slot] = nullptr; + +    generations[generation] = next_free_slot; +    next_free_slot = slot; +    return RESULT_SUCCESS;  } -void ObjectPool::List() { -    for (int i = 0; i < MAX_COUNT; i++) { -        if (occupied[i]) { -            if (pool[i]) { -                LOG_DEBUG(Kernel, "KO %i: %s \"%s\"", i + HANDLE_OFFSET, pool[i]->GetTypeName().c_str(), -                    pool[i]->GetName().c_str()); -            } -        } -    } +bool HandleTable::IsValid(Handle handle) const { +    size_t slot = GetSlot(handle); +    u16 generation = GetGeneration(handle); + +    return slot < MAX_COUNT && objects[slot] != nullptr && generations[slot] == generation;  } -int ObjectPool::GetCount() const { -    return std::count(occupied.begin(), occupied.end(), true); +Object* HandleTable::GetGeneric(Handle handle) const { +    if (handle == CurrentThread) { +        // TODO(yuriks) Directly return the pointer once this is possible. +        handle = GetCurrentThreadHandle(); +    } else if (handle == CurrentProcess) { +        LOG_ERROR(Kernel, "Current process (%08X) pseudo-handle not supported", CurrentProcess); +        return nullptr; +    } + +    if (!IsValid(handle)) { +        return nullptr; +    } +    return objects[GetSlot(handle)];  } -Object* ObjectPool::CreateByIDType(int type) { -    LOG_ERROR(Kernel, "Unimplemented: %d.", type); -    return nullptr; +void HandleTable::Clear() { +    for (size_t i = 0; i < MAX_COUNT; ++i) { +        generations[i] = i + 1; +        if (objects[i] != nullptr) +            intrusive_ptr_release(objects[i]); +        objects[i] = nullptr; +    } +    next_free_slot = 0;  }  /// Initialize the kernel @@ -95,7 +111,7 @@ void Init() {  void Shutdown() {      Kernel::ThreadingShutdown(); -    g_object_pool.Clear(); // Free all kernel objects +    g_handle_table.Clear(); // Free all kernel objects  }  /** diff --git a/src/core/hle/kernel/kernel.h b/src/core/hle/kernel/kernel.h index 32258d5a0..7f86fd07d 100644 --- a/src/core/hle/kernel/kernel.h +++ b/src/core/hle/kernel/kernel.h @@ -12,13 +12,17 @@  typedef u32 Handle;  typedef s32 Result; +const Handle INVALID_HANDLE = 0; +  namespace Kernel { -// From kernel.h. Declarations duplicated here to avoid a circular header dependency. -class Thread; -Thread* GetCurrentThread(); +// TODO: Verify code +const ResultCode ERR_OUT_OF_HANDLES(ErrorDescription::OutOfMemory, ErrorModule::Kernel, +        ErrorSummary::OutOfResource, ErrorLevel::Temporary); +// TOOD: Verify code +const ResultCode ERR_INVALID_HANDLE = InvalidHandle(ErrorModule::Kernel); -enum KernelHandle { +enum KernelHandle : Handle {      CurrentThread   = 0xFFFF8000,      CurrentProcess  = 0xFFFF8001,  }; @@ -41,10 +45,10 @@ enum {      DEFAULT_STACK_SIZE  = 0x4000,  }; -class ObjectPool; +class HandleTable;  class Object : NonCopyable { -    friend class ObjectPool; +    friend class HandleTable;      u32 handle;  public:      virtual ~Object() {} @@ -61,106 +65,130 @@ public:          LOG_ERROR(Kernel, "(UNIMPLEMENTED)");          return UnimplementedFunction(ErrorModule::Kernel);      } -}; -class ObjectPool : NonCopyable { -public: -    ObjectPool(); -    ~ObjectPool() {} +private: +    friend void intrusive_ptr_add_ref(Object*); +    friend void intrusive_ptr_release(Object*); -    // Allocates a handle within the range and inserts the object into the map. -    Handle Create(Object* obj, int range_bottom=INITIAL_NEXT_ID, int range_top=0x7FFFFFFF); +    unsigned int ref_count = 0; +}; -    static Object* CreateByIDType(int type); +// Special functions that will later be used by boost::instrusive_ptr to do automatic ref-counting +inline void intrusive_ptr_add_ref(Object* object) { +    ++object->ref_count; +} -    template <class T> -    void Destroy(Handle handle) { -        if (Get<T>(handle)) { -            occupied[handle - HANDLE_OFFSET] = false; -            delete pool[handle - HANDLE_OFFSET]; -        } +inline void intrusive_ptr_release(Object* object) { +    if (--object->ref_count == 0) { +        delete object;      } +} -    bool IsValid(Handle handle) const; +/** + * This class allows the creation of Handles, which are references to objects that can be tested + * for validity and looked up. Here they are used to pass references to kernel objects to/from the + * emulated process. it has been designed so that it follows the same handle format and has + * approximately the same restrictions as the handle manager in the CTR-OS. + * + * Handles contain two sub-fields: a slot index (bits 31:15) and a generation value (bits 14:0). + * The slot index is used to index into the arrays in this class to access the data corresponding + * to the Handle. + * + * To prevent accidental use of a freed Handle whose slot has already been reused, a global counter + * is kept and incremented every time a Handle is created. This is the Handle's "generation". The + * value of the counter is stored into the Handle as well as in the handle table (in the + * "generations" array). When looking up a handle, the Handle's generation must match with the + * value stored on the class, otherwise the Handle is considered invalid. + * + * To find free slots when allocating a Handle without needing to scan the entire object array, the + * generations field of unallocated slots is re-purposed as a linked list of indices to free slots. + * When a Handle is created, an index is popped off the list and used for the new Handle. When it + * is destroyed, it is again pushed onto the list to be re-used by the next allocation. It is + * likely that this allocation strategy differs from the one used in CTR-OS, but this hasn't been + * verified and isn't likely to cause any problems. + */ +class HandleTable final : NonCopyable { +public: +    HandleTable(); -    template <class T> -    T* Get(Handle handle) { -        if (handle == CurrentThread) { -            return reinterpret_cast<T*>(GetCurrentThread()); -        } +    /** +     * Allocates a handle for the given object. +     * @return The created Handle or one of the following errors: +     *           - `ERR_OUT_OF_HANDLES`: the maximum number of handles has been exceeded. +     */ +    ResultVal<Handle> Create(Object* obj); -        if (handle < HANDLE_OFFSET || handle >= HANDLE_OFFSET + MAX_COUNT || !occupied[handle - HANDLE_OFFSET]) { -            if (handle != 0) { -                LOG_ERROR(Kernel, "Bad object handle %08x", handle); -            } -            return nullptr; -        } else { -            Object* t = pool[handle - HANDLE_OFFSET]; -            if (t->GetHandleType() != T::GetStaticHandleType()) { -                LOG_ERROR(Kernel, "Wrong object type for %08x", handle); -                return nullptr; -            } -            return static_cast<T*>(t); -        } -    } +    /** +     * Returns a new handle that points to the same object as the passed in handle. +     * @return The duplicated Handle or one of the following errors: +     *           - `ERR_INVALID_HANDLE`: an invalid handle was passed in. +     *           - Any errors returned by `Create()`. +     */ +    ResultVal<Handle> Duplicate(Handle handle); -    // ONLY use this when you know the handle is valid. -    template <class T> -    T *GetFast(Handle handle) { -        if (handle == CurrentThread) { -            return reinterpret_cast<T*>(GetCurrentThread()); -        } +    /** +     * Closes a handle, removing it from the table and decreasing the object's ref-count. +     * @return `RESULT_SUCCESS` or one of the following errors: +     *           - `ERR_INVALID_HANDLE`: an invalid handle was passed in. +     */ +    ResultCode Close(Handle handle); -        const Handle realHandle = handle - HANDLE_OFFSET; -        _dbg_assert_(Kernel, realHandle >= 0 && realHandle < MAX_COUNT && occupied[realHandle]); -        return static_cast<T*>(pool[realHandle]); -    } +    /// Checks if a handle is valid and points to an existing object. +    bool IsValid(Handle handle) const; -    template <class T, typename ArgT> -    void Iterate(bool func(T*, ArgT), ArgT arg) { -        int type = T::GetStaticIDType(); -        for (int i = 0; i < MAX_COUNT; i++) -        { -            if (!occupied[i]) -                continue; -            T* t = static_cast<T*>(pool[i]); -            if (t->GetIDType() == type) { -                if (!func(t, arg)) -                    break; -            } -        } -    } +    /** +     * Looks up a handle. +     * @returns Pointer to the looked-up object, or `nullptr` if the handle is not valid. +     */ +    Object* GetGeneric(Handle handle) const; -    bool GetIDType(Handle handle, HandleType* type) const { -        if ((handle < HANDLE_OFFSET) || (handle >= HANDLE_OFFSET + MAX_COUNT) || -                !occupied[handle - HANDLE_OFFSET]) { -            LOG_ERROR(Kernel, "Bad object handle %08X", handle); -            return false; +    /** +     * Looks up a handle while verifying its type. +     * @returns Pointer to the looked-up object, or `nullptr` if the handle is not valid or its +     *          type differs from the handle type `T::HANDLE_TYPE`. +     */ +    template <class T> +    T* Get(Handle handle) const { +        Object* object = GetGeneric(handle); +        if (object != nullptr && object->GetHandleType() == T::HANDLE_TYPE) { +            return static_cast<T*>(object);          } -        Object* t = pool[handle - HANDLE_OFFSET]; -        *type = t->GetHandleType(); -        return true; +        return nullptr;      } -    Object* &operator [](Handle handle); -    void List(); +    /// Closes all handles held in this table.      void Clear(); -    int GetCount() const;  private: +    /** +     * This is the maximum limit of handles allowed per process in CTR-OS. It can be further +     * reduced by ExHeader values, but this is not emulated here. +     */ +    static const size_t MAX_COUNT = 4096; + +    static size_t GetSlot(Handle handle)    { return handle >> 15; } +    static u16 GetGeneration(Handle handle) { return handle & 0x7FFF; } + +    /// Stores the Object referenced by the handle or null if the slot is empty. +    std::array<Object*, MAX_COUNT> objects; -    enum { -        MAX_COUNT       = 0x1000, -        HANDLE_OFFSET   = 0x100, -        INITIAL_NEXT_ID = 0x10, -    }; +    /** +     * The value of `next_generation` when the handle was created, used to check for validity. For +     * empty slots, contains the index of the next free slot in the list. +     */ +    std::array<u16, MAX_COUNT> generations; + +    /** +     * Global counter of the number of created handles. Stored in `generations` when a handle is +     * created, and wraps around to 1 when it hits 0x8000. +     */ +    u16 next_generation; -    std::array<Object*, MAX_COUNT> pool; -    std::array<bool, MAX_COUNT> occupied; -    int next_id; +    /// Head of the free slots linked list. +    u16 next_free_slot;  }; -extern ObjectPool g_object_pool; +extern HandleTable g_handle_table;  extern Handle g_main_thread;  /// The ID code of the currently running game diff --git a/src/core/hle/kernel/mutex.cpp b/src/core/hle/kernel/mutex.cpp index 5a18af114..558068c79 100644 --- a/src/core/hle/kernel/mutex.cpp +++ b/src/core/hle/kernel/mutex.cpp @@ -18,8 +18,8 @@ public:      std::string GetTypeName() const override { return "Mutex"; }      std::string GetName() const override { return name; } -    static Kernel::HandleType GetStaticHandleType() { return Kernel::HandleType::Mutex; } -    Kernel::HandleType GetHandleType() const override { return Kernel::HandleType::Mutex; } +    static const HandleType HANDLE_TYPE = HandleType::Mutex; +    HandleType GetHandleType() const override { return HANDLE_TYPE; }      bool initial_locked;                        ///< Initial lock state when mutex was created      bool locked;                                ///< Current locked state @@ -87,7 +87,7 @@ void ReleaseThreadMutexes(Handle thread) {      // Release every mutex that the thread holds, and resume execution on the waiting threads      for (MutexMap::iterator iter = locked.first; iter != locked.second; ++iter) { -        Mutex* mutex = g_object_pool.GetFast<Mutex>(iter->second); +        Mutex* mutex = g_handle_table.Get<Mutex>(iter->second);          ResumeWaitingThread(mutex);      } @@ -115,7 +115,7 @@ bool ReleaseMutex(Mutex* mutex) {   * @param handle Handle to mutex to release   */  ResultCode ReleaseMutex(Handle handle) { -    Mutex* mutex = Kernel::g_object_pool.Get<Mutex>(handle); +    Mutex* mutex = Kernel::g_handle_table.Get<Mutex>(handle);      if (mutex == nullptr) return InvalidHandle(ErrorModule::Kernel);      if (!ReleaseMutex(mutex)) { @@ -136,7 +136,8 @@ ResultCode ReleaseMutex(Handle handle) {   */  Mutex* CreateMutex(Handle& handle, bool initial_locked, const std::string& name) {      Mutex* mutex = new Mutex; -    handle = Kernel::g_object_pool.Create(mutex); +    // TODO(yuriks): Fix error reporting +    handle = Kernel::g_handle_table.Create(mutex).ValueOr(INVALID_HANDLE);      mutex->locked = mutex->initial_locked = initial_locked;      mutex->name = name; diff --git a/src/core/hle/kernel/semaphore.cpp b/src/core/hle/kernel/semaphore.cpp index b81d0b26a..6bc8066a6 100644 --- a/src/core/hle/kernel/semaphore.cpp +++ b/src/core/hle/kernel/semaphore.cpp @@ -17,8 +17,8 @@ public:      std::string GetTypeName() const override { return "Semaphore"; }      std::string GetName() const override { return name; } -    static Kernel::HandleType GetStaticHandleType() { return Kernel::HandleType::Semaphore; } -    Kernel::HandleType GetHandleType() const override { return Kernel::HandleType::Semaphore; } +    static const HandleType HANDLE_TYPE = HandleType::Semaphore; +    HandleType GetHandleType() const override { return HANDLE_TYPE; }      s32 max_count;                              ///< Maximum number of simultaneous holders the semaphore can have      s32 available_count;                        ///< Number of free slots left in the semaphore @@ -57,7 +57,8 @@ ResultCode CreateSemaphore(Handle* handle, s32 initial_count,                            ErrorSummary::WrongArgument, ErrorLevel::Permanent);      Semaphore* semaphore = new Semaphore; -    *handle = g_object_pool.Create(semaphore); +    // TOOD(yuriks): Fix error reporting +    *handle = g_handle_table.Create(semaphore).ValueOr(INVALID_HANDLE);      // When the semaphore is created, some slots are reserved for other threads,      // and the rest is reserved for the caller thread @@ -69,7 +70,7 @@ ResultCode CreateSemaphore(Handle* handle, s32 initial_count,  }  ResultCode ReleaseSemaphore(s32* count, Handle handle, s32 release_count) { -    Semaphore* semaphore = g_object_pool.Get<Semaphore>(handle); +    Semaphore* semaphore = g_handle_table.Get<Semaphore>(handle);      if (semaphore == nullptr)          return InvalidHandle(ErrorModule::Kernel); diff --git a/src/core/hle/kernel/session.h b/src/core/hle/kernel/session.h index 6760f346e..91f3ffc2c 100644 --- a/src/core/hle/kernel/session.h +++ b/src/core/hle/kernel/session.h @@ -45,8 +45,8 @@ class Session : public Object {  public:      std::string GetTypeName() const override { return "Session"; } -    static Kernel::HandleType GetStaticHandleType() { return Kernel::HandleType::Session; } -    Kernel::HandleType GetHandleType() const override { return Kernel::HandleType::Session; } +    static const HandleType HANDLE_TYPE = HandleType::Session; +    HandleType GetHandleType() const override { return HANDLE_TYPE; }      /**       * Handles a synchronous call to this session using HLE emulation. Emulated <-> emulated calls diff --git a/src/core/hle/kernel/shared_memory.cpp b/src/core/hle/kernel/shared_memory.cpp index 2840f13bb..cea1f6fa1 100644 --- a/src/core/hle/kernel/shared_memory.cpp +++ b/src/core/hle/kernel/shared_memory.cpp @@ -13,8 +13,8 @@ class SharedMemory : public Object {  public:      std::string GetTypeName() const override { return "SharedMemory"; } -    static Kernel::HandleType GetStaticHandleType() {  return Kernel::HandleType::SharedMemory; } -    Kernel::HandleType GetHandleType() const override { return Kernel::HandleType::SharedMemory; } +    static const HandleType HANDLE_TYPE = HandleType::SharedMemory; +    HandleType GetHandleType() const override { return HANDLE_TYPE; }      u32 base_address;                   ///< Address of shared memory block in RAM      MemoryPermission permissions;       ///< Permissions of shared memory block (SVC field) @@ -32,7 +32,8 @@ public:   */  SharedMemory* CreateSharedMemory(Handle& handle, const std::string& name) {      SharedMemory* shared_memory = new SharedMemory; -    handle = Kernel::g_object_pool.Create(shared_memory); +    // TOOD(yuriks): Fix error reporting +    handle = Kernel::g_handle_table.Create(shared_memory).ValueOr(INVALID_HANDLE);      shared_memory->name = name;      return shared_memory;  } @@ -60,7 +61,7 @@ ResultCode MapSharedMemory(u32 handle, u32 address, MemoryPermission permissions          return ResultCode(ErrorDescription::InvalidAddress, ErrorModule::Kernel,                  ErrorSummary::InvalidArgument, ErrorLevel::Permanent);      } -    SharedMemory* shared_memory = Kernel::g_object_pool.Get<SharedMemory>(handle); +    SharedMemory* shared_memory = Kernel::g_handle_table.Get<SharedMemory>(handle);      if (shared_memory == nullptr) return InvalidHandle(ErrorModule::Kernel);      shared_memory->base_address = address; @@ -71,7 +72,7 @@ ResultCode MapSharedMemory(u32 handle, u32 address, MemoryPermission permissions  }  ResultVal<u8*> GetSharedMemoryPointer(Handle handle, u32 offset) { -    SharedMemory* shared_memory = Kernel::g_object_pool.Get<SharedMemory>(handle); +    SharedMemory* shared_memory = Kernel::g_handle_table.Get<SharedMemory>(handle);      if (shared_memory == nullptr) return InvalidHandle(ErrorModule::Kernel);      if (0 != shared_memory->base_address) diff --git a/src/core/hle/kernel/thread.cpp b/src/core/hle/kernel/thread.cpp index c6a8dc7b9..872df2d14 100644 --- a/src/core/hle/kernel/thread.cpp +++ b/src/core/hle/kernel/thread.cpp @@ -26,8 +26,8 @@ public:      std::string GetName() const override { return name; }      std::string GetTypeName() const override { return "Thread"; } -    static Kernel::HandleType GetStaticHandleType() { return Kernel::HandleType::Thread; } -    Kernel::HandleType GetHandleType() const override { return Kernel::HandleType::Thread; } +    static const HandleType HANDLE_TYPE = HandleType::Thread; +    HandleType GetHandleType() const override { return HANDLE_TYPE; }      inline bool IsRunning() const { return (status & THREADSTATUS_RUNNING) != 0; }      inline bool IsStopped() const { return (status & THREADSTATUS_DORMANT) != 0; } @@ -164,7 +164,7 @@ static bool CheckWaitType(const Thread* thread, WaitType type, Handle wait_handl  /// Stops the current thread  ResultCode StopThread(Handle handle, const char* reason) { -    Thread* thread = g_object_pool.Get<Thread>(handle); +    Thread* thread = g_handle_table.Get<Thread>(handle);      if (thread == nullptr) return InvalidHandle(ErrorModule::Kernel);      // Release all the mutexes that this thread holds @@ -173,7 +173,7 @@ ResultCode StopThread(Handle handle, const char* reason) {      ChangeReadyState(thread, false);      thread->status = THREADSTATUS_DORMANT;      for (Handle waiting_handle : thread->waiting_threads) { -        Thread* waiting_thread = g_object_pool.Get<Thread>(waiting_handle); +        Thread* waiting_thread = g_handle_table.Get<Thread>(waiting_handle);          if (CheckWaitType(waiting_thread, WAITTYPE_THREADEND, handle))              ResumeThreadFromWait(waiting_handle); @@ -210,7 +210,7 @@ Handle ArbitrateHighestPriorityThread(u32 arbiter, u32 address) {      // Iterate through threads, find highest priority thread that is waiting to be arbitrated...      for (Handle handle : thread_queue) { -        Thread* thread = g_object_pool.Get<Thread>(handle); +        Thread* thread = g_handle_table.Get<Thread>(handle);          if (!CheckWaitType(thread, WAITTYPE_ARB, arbiter, address))              continue; @@ -235,7 +235,7 @@ void ArbitrateAllThreads(u32 arbiter, u32 address) {      // Iterate through threads, find highest priority thread that is waiting to be arbitrated...      for (Handle handle : thread_queue) { -        Thread* thread = g_object_pool.Get<Thread>(handle); +        Thread* thread = g_handle_table.Get<Thread>(handle);          if (CheckWaitType(thread, WAITTYPE_ARB, arbiter, address))              ResumeThreadFromWait(handle); @@ -288,7 +288,7 @@ Thread* NextThread() {      if (next == 0) {          return nullptr;      } -    return Kernel::g_object_pool.Get<Thread>(next); +    return Kernel::g_handle_table.Get<Thread>(next);  }  void WaitCurrentThread(WaitType wait_type, Handle wait_handle) { @@ -305,7 +305,7 @@ void WaitCurrentThread(WaitType wait_type, Handle wait_handle, VAddr wait_addres  /// Resumes a thread from waiting by marking it as "ready"  void ResumeThreadFromWait(Handle handle) { -    Thread* thread = Kernel::g_object_pool.Get<Thread>(handle); +    Thread* thread = Kernel::g_handle_table.Get<Thread>(handle);      if (thread) {          thread->status &= ~THREADSTATUS_WAIT;          thread->wait_handle = 0; @@ -341,7 +341,8 @@ Thread* CreateThread(Handle& handle, const char* name, u32 entry_point, s32 prio      Thread* thread = new Thread; -    handle = Kernel::g_object_pool.Create(thread); +    // TOOD(yuriks): Fix error reporting +    handle = Kernel::g_handle_table.Create(thread).ValueOr(INVALID_HANDLE);      thread_queue.push_back(handle);      thread_ready_queue.prepare(priority); @@ -398,7 +399,7 @@ Handle CreateThread(const char* name, u32 entry_point, s32 priority, u32 arg, s3  /// Get the priority of the thread specified by handle  ResultVal<u32> GetThreadPriority(const Handle handle) { -    Thread* thread = g_object_pool.Get<Thread>(handle); +    Thread* thread = g_handle_table.Get<Thread>(handle);      if (thread == nullptr) return InvalidHandle(ErrorModule::Kernel);      return MakeResult<u32>(thread->current_priority); @@ -410,7 +411,7 @@ ResultCode SetThreadPriority(Handle handle, s32 priority) {      if (!handle) {          thread = GetCurrentThread(); // TODO(bunnei): Is this correct behavior?      } else { -        thread = g_object_pool.Get<Thread>(handle); +        thread = g_handle_table.Get<Thread>(handle);          if (thread == nullptr) {              return InvalidHandle(ErrorModule::Kernel);          } @@ -481,7 +482,7 @@ void Reschedule() {          LOG_TRACE(Kernel, "cannot context switch from 0x%08X, no higher priority thread!", prev->GetHandle());          for (Handle handle : thread_queue) { -            Thread* thread = g_object_pool.Get<Thread>(handle); +            Thread* thread = g_handle_table.Get<Thread>(handle);              LOG_TRACE(Kernel, "\thandle=0x%08X prio=0x%02X, status=0x%08X wait_type=0x%08X wait_handle=0x%08X",                  thread->GetHandle(), thread->current_priority, thread->status, thread->wait_type, thread->wait_handle);          } @@ -497,7 +498,7 @@ void Reschedule() {  }  ResultCode GetThreadId(u32* thread_id, Handle handle) { -    Thread* thread = g_object_pool.Get<Thread>(handle); +    Thread* thread = g_handle_table.Get<Thread>(handle);      if (thread == nullptr)          return ResultCode(ErrorDescription::InvalidHandle, ErrorModule::OS,                             ErrorSummary::WrongArgument, ErrorLevel::Permanent); diff --git a/src/core/hle/kernel/thread.h b/src/core/hle/kernel/thread.h index 9396b6b26..0e1397cd9 100644 --- a/src/core/hle/kernel/thread.h +++ b/src/core/hle/kernel/thread.h @@ -77,9 +77,6 @@ Handle ArbitrateHighestPriorityThread(u32 arbiter, u32 address);  /// Arbitrate all threads currently waiting...  void ArbitrateAllThreads(u32 arbiter, u32 address); -/// Gets the current thread -Thread* GetCurrentThread(); -  /// Gets the current thread handle  Handle GetCurrentThreadHandle(); | 
