NVMeController.h 2.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  1. /*
  2. * Copyright (c) 2021, Pankaj R <pankydev8@gmail.com>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #pragma once
  7. #include <AK/OwnPtr.h>
  8. #include <AK/Time.h>
  9. #include <AK/Tuple.h>
  10. #include <AK/Types.h>
  11. #include <Kernel/Bus/PCI/Device.h>
  12. #include <Kernel/Library/LockRefPtr.h>
  13. #include <Kernel/Library/NonnullLockRefPtr.h>
  14. #include <Kernel/Library/NonnullLockRefPtrVector.h>
  15. #include <Kernel/Locking/Spinlock.h>
  16. #include <Kernel/Memory/TypedMapping.h>
  17. #include <Kernel/Storage/NVMe/NVMeDefinitions.h>
  18. #include <Kernel/Storage/NVMe/NVMeNameSpace.h>
  19. #include <Kernel/Storage/NVMe/NVMeQueue.h>
  20. #include <Kernel/Storage/StorageController.h>
  21. namespace Kernel {
  22. class NVMeController : public PCI::Device
  23. , public StorageController {
  24. public:
  25. static ErrorOr<NonnullLockRefPtr<NVMeController>> try_initialize(PCI::DeviceIdentifier const&, bool is_queue_polled);
  26. ErrorOr<void> initialize(bool is_queue_polled);
  27. LockRefPtr<StorageDevice> device(u32 index) const override;
  28. size_t devices_count() const override;
  29. virtual StringView device_name() const override { return "NVMeController"sv; }
  30. protected:
  31. bool reset() override;
  32. bool shutdown() override;
  33. void complete_current_request(AsyncDeviceRequest::RequestResult result) override;
  34. public:
  35. bool reset_controller();
  36. bool start_controller();
  37. u32 get_admin_q_dept();
  38. u16 submit_admin_command(NVMeSubmission& sub, bool sync = false)
  39. {
  40. // First queue is always the admin queue
  41. if (sync) {
  42. return m_admin_queue->submit_sync_sqe(sub);
  43. }
  44. m_admin_queue->submit_sqe(sub);
  45. return 0;
  46. }
  47. bool is_admin_queue_ready() { return m_admin_queue_ready; };
  48. void set_admin_queue_ready_flag() { m_admin_queue_ready = true; };
  49. private:
  50. NVMeController(PCI::DeviceIdentifier const&, u32 hardware_relative_controller_id);
  51. ErrorOr<void> identify_and_init_namespaces();
  52. Tuple<u64, u8> get_ns_features(IdentifyNamespace& identify_data_struct);
  53. ErrorOr<void> create_admin_queue(Optional<u8> irq);
  54. ErrorOr<void> create_io_queue(u8 qid, Optional<u8> irq);
  55. void calculate_doorbell_stride()
  56. {
  57. m_dbl_stride = (m_controller_regs->cap >> CAP_DBL_SHIFT) & CAP_DBL_MASK;
  58. }
  59. bool wait_for_ready(bool);
  60. private:
  61. LockRefPtr<NVMeQueue> m_admin_queue;
  62. NonnullLockRefPtrVector<NVMeQueue> m_queues;
  63. NonnullLockRefPtrVector<NVMeNameSpace> m_namespaces;
  64. Memory::TypedMapping<ControllerRegister volatile> m_controller_regs;
  65. bool m_admin_queue_ready { false };
  66. size_t m_device_count { 0 };
  67. AK::Time m_ready_timeout;
  68. u32 m_bar { 0 };
  69. u8 m_dbl_stride { 0 };
  70. static Atomic<u8> s_controller_id;
  71. };
  72. }