mirror of
https://github.com/RGBCube/serenity
synced 2025-07-25 19:37:35 +00:00
Kernel: Add Device base class for CharacterDevice.
..to prepare for adding a BlockDevice class.
This commit is contained in:
parent
c6ca6522fc
commit
994279d56c
11 changed files with 113 additions and 94 deletions
|
@ -1,20 +1,7 @@
|
||||||
#include "CharacterDevice.h"
|
#pragma once
|
||||||
#include <LibC/errno_numbers.h>
|
|
||||||
|
#include <Kernel/CharacterDevice.h>
|
||||||
|
|
||||||
CharacterDevice::~CharacterDevice()
|
CharacterDevice::~CharacterDevice()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
RetainPtr<FileDescriptor> CharacterDevice::open(int& error, int options)
|
|
||||||
{
|
|
||||||
return VFS::the().open(*this, error, options);
|
|
||||||
}
|
|
||||||
|
|
||||||
void CharacterDevice::close()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
int CharacterDevice::ioctl(Process&, unsigned, unsigned)
|
|
||||||
{
|
|
||||||
return -ENOTTY;
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,48 +1,11 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <AK/Retainable.h>
|
#include <Kernel/Device.h>
|
||||||
#include <AK/Types.h>
|
|
||||||
#include "Limits.h"
|
|
||||||
#include "FileDescriptor.h"
|
|
||||||
|
|
||||||
class Process;
|
class CharacterDevice : public Device {
|
||||||
|
|
||||||
class CharacterDevice : public Retainable<CharacterDevice> {
|
|
||||||
public:
|
public:
|
||||||
virtual ~CharacterDevice();
|
virtual ~CharacterDevice() override;
|
||||||
|
|
||||||
InodeMetadata metadata() const { return { }; }
|
|
||||||
|
|
||||||
virtual RetainPtr<FileDescriptor> open(int& error, int options);
|
|
||||||
virtual void close();
|
|
||||||
|
|
||||||
virtual bool can_read(Process&) const = 0;
|
|
||||||
virtual bool can_write(Process&) const = 0;
|
|
||||||
|
|
||||||
virtual ssize_t read(Process&, byte* buffer, size_t bufferSize) = 0;
|
|
||||||
virtual ssize_t write(Process&, const byte* buffer, size_t bufferSize) = 0;
|
|
||||||
|
|
||||||
unsigned major() const { return m_major; }
|
|
||||||
unsigned minor() const { return m_minor; }
|
|
||||||
|
|
||||||
virtual bool is_tty() const { return false; }
|
|
||||||
virtual bool is_master_pty() const { return false; }
|
|
||||||
|
|
||||||
virtual int ioctl(Process&, unsigned request, unsigned arg);
|
|
||||||
|
|
||||||
virtual const char* class_name() const = 0;
|
|
||||||
|
|
||||||
uid_t uid() const { return m_uid; }
|
|
||||||
uid_t gid() const { return m_gid; }
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
CharacterDevice(unsigned major, unsigned minor) : m_major(major), m_minor(minor) { }
|
CharacterDevice(unsigned major, unsigned minor) : Device(major, minor) { }
|
||||||
void set_uid(uid_t uid) { m_uid = uid; }
|
|
||||||
void set_gid(gid_t gid) { m_gid = gid; }
|
|
||||||
|
|
||||||
private:
|
|
||||||
unsigned m_major { 0 };
|
|
||||||
unsigned m_minor { 0 };
|
|
||||||
uid_t m_uid { 0 };
|
|
||||||
gid_t m_gid { 0 };
|
|
||||||
};
|
};
|
||||||
|
|
20
Kernel/Device.cpp
Normal file
20
Kernel/Device.cpp
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
#include "CharacterDevice.h"
|
||||||
|
#include <LibC/errno_numbers.h>
|
||||||
|
|
||||||
|
Device::~Device()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
RetainPtr<FileDescriptor> Device::open(int& error, int options)
|
||||||
|
{
|
||||||
|
return VFS::the().open(*this, error, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Device::close()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
int Device::ioctl(Process&, unsigned, unsigned)
|
||||||
|
{
|
||||||
|
return -ENOTTY;
|
||||||
|
}
|
48
Kernel/Device.h
Normal file
48
Kernel/Device.h
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <AK/Retainable.h>
|
||||||
|
#include <AK/Types.h>
|
||||||
|
#include "Limits.h"
|
||||||
|
#include "FileDescriptor.h"
|
||||||
|
|
||||||
|
class Process;
|
||||||
|
|
||||||
|
class Device : public Retainable<Device> {
|
||||||
|
public:
|
||||||
|
virtual ~Device();
|
||||||
|
|
||||||
|
InodeMetadata metadata() const { return { }; }
|
||||||
|
|
||||||
|
virtual RetainPtr<FileDescriptor> open(int& error, int options);
|
||||||
|
virtual void close();
|
||||||
|
|
||||||
|
virtual bool can_read(Process&) const = 0;
|
||||||
|
virtual bool can_write(Process&) const = 0;
|
||||||
|
|
||||||
|
virtual ssize_t read(Process&, byte* buffer, size_t bufferSize) = 0;
|
||||||
|
virtual ssize_t write(Process&, const byte* buffer, size_t bufferSize) = 0;
|
||||||
|
|
||||||
|
unsigned major() const { return m_major; }
|
||||||
|
unsigned minor() const { return m_minor; }
|
||||||
|
|
||||||
|
virtual bool is_tty() const { return false; }
|
||||||
|
virtual bool is_master_pty() const { return false; }
|
||||||
|
|
||||||
|
virtual int ioctl(Process&, unsigned request, unsigned arg);
|
||||||
|
|
||||||
|
virtual const char* class_name() const = 0;
|
||||||
|
|
||||||
|
uid_t uid() const { return m_uid; }
|
||||||
|
uid_t gid() const { return m_gid; }
|
||||||
|
|
||||||
|
protected:
|
||||||
|
Device(unsigned major, unsigned minor) : m_major(major), m_minor(minor) { }
|
||||||
|
void set_uid(uid_t uid) { m_uid = uid; }
|
||||||
|
void set_gid(gid_t gid) { m_gid = gid; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
unsigned m_major { 0 };
|
||||||
|
unsigned m_minor { 0 };
|
||||||
|
uid_t m_uid { 0 };
|
||||||
|
gid_t m_gid { 0 };
|
||||||
|
};
|
|
@ -14,7 +14,7 @@ RetainPtr<FileDescriptor> FileDescriptor::create(RetainPtr<Inode>&& inode)
|
||||||
return adopt(*new FileDescriptor(move(inode)));
|
return adopt(*new FileDescriptor(move(inode)));
|
||||||
}
|
}
|
||||||
|
|
||||||
RetainPtr<FileDescriptor> FileDescriptor::create(RetainPtr<CharacterDevice>&& device)
|
RetainPtr<FileDescriptor> FileDescriptor::create(RetainPtr<Device>&& device)
|
||||||
{
|
{
|
||||||
return adopt(*new FileDescriptor(move(device)));
|
return adopt(*new FileDescriptor(move(device)));
|
||||||
}
|
}
|
||||||
|
@ -39,7 +39,7 @@ FileDescriptor::FileDescriptor(RetainPtr<Inode>&& inode)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
FileDescriptor::FileDescriptor(RetainPtr<CharacterDevice>&& device)
|
FileDescriptor::FileDescriptor(RetainPtr<Device>&& device)
|
||||||
: m_device(move(device))
|
: m_device(move(device))
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,7 +18,7 @@ public:
|
||||||
|
|
||||||
static RetainPtr<FileDescriptor> create(RetainPtr<Socket>&&, SocketRole = SocketRole::None);
|
static RetainPtr<FileDescriptor> create(RetainPtr<Socket>&&, SocketRole = SocketRole::None);
|
||||||
static RetainPtr<FileDescriptor> create(RetainPtr<Inode>&&);
|
static RetainPtr<FileDescriptor> create(RetainPtr<Inode>&&);
|
||||||
static RetainPtr<FileDescriptor> create(RetainPtr<CharacterDevice>&&);
|
static RetainPtr<FileDescriptor> create(RetainPtr<Device>&&);
|
||||||
static RetainPtr<FileDescriptor> create_pipe_writer(FIFO&);
|
static RetainPtr<FileDescriptor> create_pipe_writer(FIFO&);
|
||||||
static RetainPtr<FileDescriptor> create_pipe_reader(FIFO&);
|
static RetainPtr<FileDescriptor> create_pipe_reader(FIFO&);
|
||||||
~FileDescriptor();
|
~FileDescriptor();
|
||||||
|
@ -44,8 +44,8 @@ public:
|
||||||
bool is_directory() const;
|
bool is_directory() const;
|
||||||
|
|
||||||
bool is_character_device() const { return m_device.ptr(); }
|
bool is_character_device() const { return m_device.ptr(); }
|
||||||
CharacterDevice* character_device() { return m_device.ptr(); }
|
Device* character_device() { return m_device.ptr(); }
|
||||||
const CharacterDevice* character_device() const { return m_device.ptr(); }
|
const Device* character_device() const { return m_device.ptr(); }
|
||||||
|
|
||||||
bool is_tty() const;
|
bool is_tty() const;
|
||||||
const TTY* tty() const;
|
const TTY* tty() const;
|
||||||
|
@ -84,11 +84,11 @@ private:
|
||||||
friend class VFS;
|
friend class VFS;
|
||||||
FileDescriptor(RetainPtr<Socket>&&, SocketRole);
|
FileDescriptor(RetainPtr<Socket>&&, SocketRole);
|
||||||
explicit FileDescriptor(RetainPtr<Inode>&&);
|
explicit FileDescriptor(RetainPtr<Inode>&&);
|
||||||
explicit FileDescriptor(RetainPtr<CharacterDevice>&&);
|
explicit FileDescriptor(RetainPtr<Device>&&);
|
||||||
FileDescriptor(FIFO&, FIFO::Direction);
|
FileDescriptor(FIFO&, FIFO::Direction);
|
||||||
|
|
||||||
RetainPtr<Inode> m_inode;
|
RetainPtr<Inode> m_inode;
|
||||||
RetainPtr<CharacterDevice> m_device;
|
RetainPtr<Device> m_device;
|
||||||
|
|
||||||
off_t m_current_offset { 0 };
|
off_t m_current_offset { 0 };
|
||||||
|
|
||||||
|
|
|
@ -37,6 +37,7 @@ KERNEL_OBJS = \
|
||||||
|
|
||||||
VFS_OBJS = \
|
VFS_OBJS = \
|
||||||
DiskDevice.o \
|
DiskDevice.o \
|
||||||
|
Device.o \
|
||||||
CharacterDevice.o \
|
CharacterDevice.o \
|
||||||
NullDevice.o \
|
NullDevice.o \
|
||||||
FullDevice.o \
|
FullDevice.o \
|
||||||
|
|
|
@ -10,7 +10,7 @@ SlavePTY::SlavePTY(MasterPTY& master, unsigned index)
|
||||||
{
|
{
|
||||||
set_uid(current->uid());
|
set_uid(current->uid());
|
||||||
set_gid(current->gid());
|
set_gid(current->gid());
|
||||||
VFS::the().register_character_device(*this);
|
VFS::the().register_device(*this);
|
||||||
DevPtsFS::the().register_slave_pty(*this);
|
DevPtsFS::the().register_slave_pty(*this);
|
||||||
set_size(80, 25);
|
set_size(80, 25);
|
||||||
}
|
}
|
||||||
|
@ -19,7 +19,7 @@ SlavePTY::~SlavePTY()
|
||||||
{
|
{
|
||||||
dbgprintf("~SlavePTY(%u)\n", m_index);
|
dbgprintf("~SlavePTY(%u)\n", m_index);
|
||||||
DevPtsFS::the().unregister_slave_pty(*this);
|
DevPtsFS::the().unregister_slave_pty(*this);
|
||||||
VFS::the().unregister_character_device(*this);
|
VFS::the().unregister_device(*this);
|
||||||
}
|
}
|
||||||
|
|
||||||
String SlavePTY::tty_name() const
|
String SlavePTY::tty_name() const
|
||||||
|
|
|
@ -122,7 +122,7 @@ void VFS::traverse_directory_inode(Inode& dir_inode, Function<bool(const FS::Dir
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
RetainPtr<FileDescriptor> VFS::open(RetainPtr<CharacterDevice>&& device, int& error, int options)
|
RetainPtr<FileDescriptor> VFS::open(RetainPtr<Device>&& device, int& error, int options)
|
||||||
{
|
{
|
||||||
// FIXME: Respect options.
|
// FIXME: Respect options.
|
||||||
(void) options;
|
(void) options;
|
||||||
|
@ -146,9 +146,9 @@ RetainPtr<FileDescriptor> VFS::open(const String& path, int& error, int options,
|
||||||
return nullptr;
|
return nullptr;
|
||||||
auto metadata = inode->metadata();
|
auto metadata = inode->metadata();
|
||||||
if (!(options & O_DONT_OPEN_DEVICE) && metadata.is_character_device()) {
|
if (!(options & O_DONT_OPEN_DEVICE) && metadata.is_character_device()) {
|
||||||
auto it = m_character_devices.find(encoded_device(metadata.major_device, metadata.minor_device));
|
auto it = m_devices.find(encoded_device(metadata.major_device, metadata.minor_device));
|
||||||
if (it == m_character_devices.end()) {
|
if (it == m_devices.end()) {
|
||||||
kprintf("VFS::open: no such character device %u,%u\n", metadata.major_device, metadata.minor_device);
|
kprintf("VFS::open: no such device %u,%u\n", metadata.major_device, metadata.minor_device);
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
auto descriptor = (*it).value->open(error, options);
|
auto descriptor = (*it).value->open(error, options);
|
||||||
|
@ -516,20 +516,20 @@ VFS::Mount::Mount(InodeIdentifier host, RetainPtr<FS>&& guest_fs)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
void VFS::register_character_device(CharacterDevice& device)
|
void VFS::register_device(Device& device)
|
||||||
{
|
{
|
||||||
m_character_devices.set(encoded_device(device.major(), device.minor()), &device);
|
m_devices.set(encoded_device(device.major(), device.minor()), &device);
|
||||||
}
|
}
|
||||||
|
|
||||||
void VFS::unregister_character_device(CharacterDevice& device)
|
void VFS::unregister_device(Device& device)
|
||||||
{
|
{
|
||||||
m_character_devices.remove(encoded_device(device.major(), device.minor()));
|
m_devices.remove(encoded_device(device.major(), device.minor()));
|
||||||
}
|
}
|
||||||
|
|
||||||
CharacterDevice* VFS::get_device(unsigned major, unsigned minor)
|
Device* VFS::get_device(unsigned major, unsigned minor)
|
||||||
{
|
{
|
||||||
auto it = m_character_devices.find(encoded_device(major, minor));
|
auto it = m_devices.find(encoded_device(major, minor));
|
||||||
if (it == m_character_devices.end())
|
if (it == m_devices.end())
|
||||||
return nullptr;
|
return nullptr;
|
||||||
return (*it).value;
|
return (*it).value;
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@
|
||||||
#define O_NOFOLLOW_NOERROR 0x4000000
|
#define O_NOFOLLOW_NOERROR 0x4000000
|
||||||
#define O_DONT_OPEN_DEVICE 0x8000000
|
#define O_DONT_OPEN_DEVICE 0x8000000
|
||||||
|
|
||||||
class CharacterDevice;
|
class Device;
|
||||||
class FileDescriptor;
|
class FileDescriptor;
|
||||||
|
|
||||||
inline constexpr dword encoded_device(unsigned major, unsigned minor)
|
inline constexpr dword encoded_device(unsigned major, unsigned minor)
|
||||||
|
@ -62,7 +62,7 @@ public:
|
||||||
bool mount_root(RetainPtr<FS>&&);
|
bool mount_root(RetainPtr<FS>&&);
|
||||||
bool mount(RetainPtr<FS>&&, const String& path);
|
bool mount(RetainPtr<FS>&&, const String& path);
|
||||||
|
|
||||||
RetainPtr<FileDescriptor> open(RetainPtr<CharacterDevice>&&, int& error, int options);
|
RetainPtr<FileDescriptor> open(RetainPtr<Device>&&, int& error, int options);
|
||||||
RetainPtr<FileDescriptor> open(const String& path, int& error, int options, mode_t mode, Inode& base);
|
RetainPtr<FileDescriptor> open(const String& path, int& error, int options, mode_t mode, Inode& base);
|
||||||
RetainPtr<FileDescriptor> create(const String& path, int& error, int options, mode_t mode, Inode& base);
|
RetainPtr<FileDescriptor> create(const String& path, int& error, int options, mode_t mode, Inode& base);
|
||||||
bool mkdir(const String& path, mode_t mode, Inode& base, int& error);
|
bool mkdir(const String& path, mode_t mode, Inode& base, int& error);
|
||||||
|
@ -70,8 +70,8 @@ public:
|
||||||
bool rmdir(const String& path, Inode& base, int& error);
|
bool rmdir(const String& path, Inode& base, int& error);
|
||||||
bool chmod(const String& path, mode_t, Inode& base, int& error);
|
bool chmod(const String& path, mode_t, Inode& base, int& error);
|
||||||
|
|
||||||
void register_character_device(CharacterDevice&);
|
void register_device(Device&);
|
||||||
void unregister_character_device(CharacterDevice&);
|
void unregister_device(Device&);
|
||||||
|
|
||||||
size_t mount_count() const { return m_mounts.size(); }
|
size_t mount_count() const { return m_mounts.size(); }
|
||||||
void for_each_mount(Function<void(const Mount&)>) const;
|
void for_each_mount(Function<void(const Mount&)>) const;
|
||||||
|
@ -85,7 +85,7 @@ public:
|
||||||
|
|
||||||
void sync();
|
void sync();
|
||||||
|
|
||||||
CharacterDevice* get_device(unsigned major, unsigned minor);
|
Device* get_device(unsigned major, unsigned minor);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
friend class FileDescriptor;
|
friend class FileDescriptor;
|
||||||
|
@ -103,6 +103,6 @@ private:
|
||||||
|
|
||||||
RetainPtr<Inode> m_root_inode;
|
RetainPtr<Inode> m_root_inode;
|
||||||
Vector<OwnPtr<Mount>> m_mounts;
|
Vector<OwnPtr<Mount>> m_mounts;
|
||||||
HashMap<dword, CharacterDevice*> m_character_devices;
|
HashMap<dword, Device*> m_devices;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -66,25 +66,25 @@ VFS* vfs;
|
||||||
Syscall::initialize();
|
Syscall::initialize();
|
||||||
|
|
||||||
auto dev_zero = make<ZeroDevice>();
|
auto dev_zero = make<ZeroDevice>();
|
||||||
vfs->register_character_device(*dev_zero);
|
vfs->register_device(*dev_zero);
|
||||||
|
|
||||||
vfs->register_character_device(*dev_null);
|
vfs->register_device(*dev_null);
|
||||||
|
|
||||||
auto dev_full = make<FullDevice>();
|
auto dev_full = make<FullDevice>();
|
||||||
vfs->register_character_device(*dev_full);
|
vfs->register_device(*dev_full);
|
||||||
|
|
||||||
auto dev_random = make<RandomDevice>();
|
auto dev_random = make<RandomDevice>();
|
||||||
vfs->register_character_device(*dev_random);
|
vfs->register_device(*dev_random);
|
||||||
|
|
||||||
auto dev_ptmx = make<PTYMultiplexer>();
|
auto dev_ptmx = make<PTYMultiplexer>();
|
||||||
vfs->register_character_device(*dev_ptmx);
|
vfs->register_device(*dev_ptmx);
|
||||||
|
|
||||||
vfs->register_character_device(*keyboard);
|
vfs->register_device(*keyboard);
|
||||||
vfs->register_character_device(*ps2mouse);
|
vfs->register_device(*ps2mouse);
|
||||||
vfs->register_character_device(*tty0);
|
vfs->register_device(*tty0);
|
||||||
vfs->register_character_device(*tty1);
|
vfs->register_device(*tty1);
|
||||||
vfs->register_character_device(*tty2);
|
vfs->register_device(*tty2);
|
||||||
vfs->register_character_device(*tty3);
|
vfs->register_device(*tty3);
|
||||||
|
|
||||||
auto dev_hd0 = IDEDiskDevice::create();
|
auto dev_hd0 = IDEDiskDevice::create();
|
||||||
auto e2fs = Ext2FS::create(dev_hd0.copy_ref());
|
auto e2fs = Ext2FS::create(dev_hd0.copy_ref());
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue