/* * Copyright (c) 2021, Liav A. * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Kernel { class AsyncBlockDeviceRequest; class AHCIPortHandler; class SATADiskDevice; class AHCIPort : public RefCounted { friend class AHCIPortHandler; friend class SATADiskDevice; public: UNMAP_AFTER_INIT static NonnullRefPtr create(const AHCIPortHandler&, volatile AHCI::PortRegisters&, u32 port_index); u32 port_index() const { return m_port_index; } u32 representative_port_index() const { return port_index() + 1; } bool is_operable() const; bool is_hot_pluggable() const; bool is_atapi_attached() const { return m_port_registers.sig == (u32)AHCI::DeviceSignature::ATAPI; }; RefPtr connected_device() const { return m_connected_device; } bool reset(); UNMAP_AFTER_INIT bool initialize_without_reset(); void handle_interrupt(); private: bool is_phy_enabled() const { return (m_port_registers.ssts & 0xf) == 3; } bool initialize(SpinlockLocker>&); UNMAP_AFTER_INIT AHCIPort(const AHCIPortHandler&, volatile AHCI::PortRegisters&, u32 port_index); ALWAYS_INLINE void clear_sata_error_register() const; void eject(); const char* try_disambiguate_sata_status(); void try_disambiguate_sata_error(); bool initiate_sata_reset(SpinlockLocker>&); void rebase(); void recover_from_fatal_error(); bool shutdown(); ALWAYS_INLINE void spin_up() const; ALWAYS_INLINE void power_on() const; void start_request(AsyncBlockDeviceRequest&); void complete_current_request(AsyncDeviceRequest::RequestResult); bool access_device(AsyncBlockDeviceRequest::RequestType, u64 lba, u8 block_count); size_t calculate_descriptors_count(size_t block_count) const; [[nodiscard]] Optional prepare_and_set_scatter_list(AsyncBlockDeviceRequest& request); ALWAYS_INLINE bool is_interrupts_enabled() const; bool spin_until_ready() const; bool identify_device(SpinlockLocker>&); ALWAYS_INLINE void start_command_list_processing() const; ALWAYS_INLINE void mark_command_header_ready_to_process(u8 command_header_index) const; ALWAYS_INLINE void stop_command_list_processing() const; ALWAYS_INLINE void start_fis_receiving() const; ALWAYS_INLINE void stop_fis_receiving() const; ALWAYS_INLINE void set_active_state() const; ALWAYS_INLINE void set_sleep_state() const; void set_interface_state(AHCI::DeviceDetectionInitialization); Optional try_to_find_unused_command_header(); ALWAYS_INLINE bool is_interface_disabled() const { return (m_port_registers.ssts & 0xf) == 4; }; // Data members EntropySource m_entropy_source; RefPtr m_current_request; Spinlock m_hard_lock; Mutex m_lock { "AHCIPort" }; mutable bool m_wait_for_completion { false }; bool m_wait_connect_for_completion { false }; NonnullRefPtrVector m_dma_buffers; NonnullRefPtrVector m_command_table_pages; RefPtr m_command_list_page; OwnPtr m_command_list_region; RefPtr m_fis_receive_page; RefPtr m_connected_device; u32 m_port_index; volatile AHCI::PortRegisters& m_port_registers; NonnullRefPtr m_parent_handler; AHCI::PortInterruptStatusBitField m_interrupt_status; AHCI::PortInterruptEnableBitField m_interrupt_enable; RefPtr m_current_scatter_list; bool m_disabled_by_firmware { false }; }; }