mirror of
				https://github.com/Atmosphere-NX/Atmosphere-libs.git
				synced 2025-10-26 09:25:48 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			142 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			142 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2020 Atmosphère-NX
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms and conditions of the GNU General Public License,
 | |
|  * version 2, as published by the Free Software Foundation.
 | |
|  *
 | |
|  * This program is distributed in the hope it will be useful, but WITHOUT
 | |
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 | |
|  * more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| #pragma once
 | |
| #include <vapours.hpp>
 | |
| #include <mesosphere/kern_select_cpu.hpp>
 | |
| #include <mesosphere/kern_k_spin_lock.hpp>
 | |
| #include <mesosphere/kern_k_interrupt_task.hpp>
 | |
| #include <mesosphere/kern_select_interrupt_controller.hpp>
 | |
| 
 | |
| namespace ams::kern::arch::arm64 {
 | |
| 
 | |
|     class KInterruptManager {
 | |
|         NON_COPYABLE(KInterruptManager);
 | |
|         NON_MOVEABLE(KInterruptManager);
 | |
|         private:
 | |
|             struct KCoreLocalInterruptEntry {
 | |
|                 KInterruptHandler *handler;
 | |
|                 bool manually_cleared;
 | |
|                 bool needs_clear;
 | |
|                 u8 priority;
 | |
| 
 | |
|                 constexpr KCoreLocalInterruptEntry()
 | |
|                     : handler(nullptr), manually_cleared(false), needs_clear(false), priority(KInterruptController::PriorityLevel_Low)
 | |
|                 {
 | |
|                     /* ... */
 | |
|                 }
 | |
|             };
 | |
| 
 | |
|             struct KGlobalInterruptEntry {
 | |
|                 KInterruptHandler *handler;
 | |
|                 bool manually_cleared;
 | |
|                 bool needs_clear;
 | |
| 
 | |
|                 constexpr KGlobalInterruptEntry() : handler(nullptr), manually_cleared(false), needs_clear(false) { /* ... */ }
 | |
|             };
 | |
|         private:
 | |
|             static KSpinLock s_lock;
 | |
|             static std::array<KGlobalInterruptEntry, KInterruptController::NumGlobalInterrupts> s_global_interrupts;
 | |
|             static KInterruptController::GlobalState s_global_state;
 | |
|             static bool s_global_state_saved;
 | |
|         private:
 | |
|             KCoreLocalInterruptEntry core_local_interrupts[KInterruptController::NumLocalInterrupts];
 | |
|             KInterruptController interrupt_controller;
 | |
|             KInterruptController::LocalState local_state;
 | |
|             bool local_state_saved;
 | |
|         private:
 | |
|             static ALWAYS_INLINE KSpinLock &GetLock() { return s_lock; }
 | |
|             static ALWAYS_INLINE KGlobalInterruptEntry &GetGlobalInterruptEntry(s32 irq) { return s_global_interrupts[KInterruptController::GetGlobalInterruptIndex(irq)]; }
 | |
|             ALWAYS_INLINE KCoreLocalInterruptEntry &GetLocalInterruptEntry(s32 irq) { return this->core_local_interrupts[KInterruptController::GetLocalInterruptIndex(irq)]; }
 | |
| 
 | |
|             bool OnHandleInterrupt();
 | |
|         public:
 | |
|             constexpr KInterruptManager() : core_local_interrupts(), interrupt_controller(), local_state(), local_state_saved(false) { /* ... */ }
 | |
|             NOINLINE void Initialize(s32 core_id);
 | |
|             NOINLINE void Finalize(s32 core_id);
 | |
| 
 | |
|             NOINLINE void Save(s32 core_id);
 | |
|             NOINLINE void Restore(s32 core_id);
 | |
| 
 | |
|             bool IsInterruptDefined(s32 irq) const {
 | |
|                 return this->interrupt_controller.IsInterruptDefined(irq);
 | |
|             }
 | |
| 
 | |
|             bool IsGlobal(s32 irq) const {
 | |
|                 return this->interrupt_controller.IsGlobal(irq);
 | |
|             }
 | |
| 
 | |
|             bool IsLocal(s32 irq) const {
 | |
|                 return this->interrupt_controller.IsLocal(irq);
 | |
|             }
 | |
| 
 | |
|             NOINLINE Result BindHandler(KInterruptHandler *handler, s32 irq, s32 core_id, s32 priority, bool manual_clear, bool level);
 | |
|             NOINLINE Result UnbindHandler(s32 irq, s32 core);
 | |
| 
 | |
|             NOINLINE Result ClearInterrupt(s32 irq);
 | |
|             NOINLINE Result ClearInterrupt(s32 irq, s32 core_id);
 | |
| 
 | |
|             ALWAYS_INLINE void SendInterProcessorInterrupt(s32 irq, u64 core_mask) {
 | |
|                 this->interrupt_controller.SendInterProcessorInterrupt(irq, core_mask);
 | |
|             }
 | |
| 
 | |
|             ALWAYS_INLINE void SendInterProcessorInterrupt(s32 irq) {
 | |
|                 this->interrupt_controller.SendInterProcessorInterrupt(irq);
 | |
|             }
 | |
| 
 | |
|             static void HandleInterrupt(bool user_mode);
 | |
| 
 | |
|             /* Implement more KInterruptManager functionality. */
 | |
|         private:
 | |
|             Result BindGlobal(KInterruptHandler *handler, s32 irq, s32 core_id, s32 priority, bool manual_clear, bool level);
 | |
|             Result BindLocal(KInterruptHandler *handler, s32 irq, s32 priority, bool manual_clear);
 | |
|             Result UnbindGlobal(s32 irq);
 | |
|             Result UnbindLocal(s32 irq);
 | |
|             Result ClearGlobal(s32 irq);
 | |
|             Result ClearLocal(s32 irq);
 | |
|         public:
 | |
|             static ALWAYS_INLINE u32 DisableInterrupts() {
 | |
|                 u64 intr_state;
 | |
|                 __asm__ __volatile__("mrs %[intr_state], daif\n"
 | |
|                                      "msr daifset, #2"
 | |
|                                      : [intr_state]"=r"(intr_state)
 | |
|                                      :: "memory");
 | |
|                 return intr_state;
 | |
|             }
 | |
| 
 | |
|             static ALWAYS_INLINE u32 EnableInterrupts() {
 | |
|                 u64 intr_state;
 | |
|                 __asm__ __volatile__("mrs %[intr_state], daif\n"
 | |
|                                      "msr daifclr, #2"
 | |
|                                      : [intr_state]"=r"(intr_state)
 | |
|                                      :: "memory");
 | |
|                 return intr_state;
 | |
|             }
 | |
| 
 | |
|             static ALWAYS_INLINE void RestoreInterrupts(u32 intr_state) {
 | |
|                 u64 cur_state;
 | |
|                 __asm__ __volatile__("mrs %[cur_state], daif" : [cur_state]"=r"(cur_state));
 | |
|                 __asm__ __volatile__("msr daif, %[intr_state]" :: [intr_state]"r"((cur_state & ~0x80ul) | (intr_state & 0x80)));
 | |
|             }
 | |
| 
 | |
|             static ALWAYS_INLINE bool AreInterruptsEnabled() {
 | |
|                 u64 intr_state;
 | |
|                 __asm__ __volatile__("mrs %[intr_state], daif" : [intr_state]"=r"(intr_state));
 | |
|                 return (intr_state & 0x80) == 0;
 | |
|             }
 | |
|     };
 | |
| 
 | |
| }
 |