Newer
Older
minerva / Kernel / Arch / aarch64 / RPi / MiniUART.h
@minerva minerva on 13 Jul 1 KB Initial commit
/*
 * Copyright (c) 2023, Daniel Bertalan <dani@danielbertalan.dev>
 *
 * SPDX-License-Identifier: BSD-2-Clause
 */

#pragma once

#include <Kernel/Devices/CharacterDevice.h>
#include <Kernel/Devices/Device.h>
#include <Kernel/Firmware/DeviceTree/Device.h>
#include <Kernel/Locking/Spinlock.h>
#include <Kernel/Memory/TypedMapping.h>

namespace Kernel::RPi {

struct MiniUARTRegisters;

// Makes the secondary "mini UART" (UART1) available to the userspace.
// See bcm2711-peripherals.pdf chapter "2.2. Mini UART".
class MiniUART final : public CharacterDevice {
    friend class Kernel::Device;

public:
    static ErrorOr<NonnullRefPtr<MiniUART>> create(DeviceTree::Device::Resource registers_resource);

    virtual ~MiniUART() override;

    // ^CharacterDevice
    virtual bool can_read(OpenFileDescription const&, u64) const override;
    virtual ErrorOr<size_t> read(OpenFileDescription&, u64, UserOrKernelBuffer&, size_t) override;
    virtual bool can_write(OpenFileDescription const&, u64) const override;
    virtual ErrorOr<size_t> write(OpenFileDescription&, u64, UserOrKernelBuffer const&, size_t) override;

    void put_char(u8);

private:
    MiniUART(Memory::TypedMapping<MiniUARTRegisters volatile>);

    // ^CharacterDevice
    virtual StringView class_name() const override { return "MiniUART"sv; }

    void set_baud_rate(u32);

    bool m_last_put_char_was_carriage_return { false };
    Spinlock<LockRank::None> m_serial_lock {};
    Memory::TypedMapping<MiniUARTRegisters volatile> m_registers;
};
}