forked from Bananymous/banan-os
Kernel: Expose /proc/meminfo and fix /proc/<pid>/meminfo
Thread was unconditionally calling <stack>->size() without validating that the stack actually exists
This commit is contained in:
parent
2a659a9d03
commit
44f0ec601f
|
@ -10,7 +10,7 @@ namespace Kernel
|
||||||
class ProcPidInode final : public TmpDirectoryInode
|
class ProcPidInode final : public TmpDirectoryInode
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
static BAN::ErrorOr<BAN::RefPtr<ProcPidInode>> create_new(Process&, TmpFileSystem&, mode_t, uid_t, gid_t);
|
static BAN::ErrorOr<BAN::RefPtr<ProcPidInode>> create_new(Process&, TmpFileSystem&, mode_t);
|
||||||
~ProcPidInode() = default;
|
~ProcPidInode() = default;
|
||||||
|
|
||||||
virtual uid_t uid() const override { return m_process.credentials().ruid(); }
|
virtual uid_t uid() const override { return m_process.credentials().ruid(); }
|
||||||
|
@ -28,11 +28,11 @@ namespace Kernel
|
||||||
Process& m_process;
|
Process& m_process;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ProcROInode final : public TmpInode
|
class ProcROProcessInode final : public TmpInode
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
static BAN::ErrorOr<BAN::RefPtr<ProcROInode>> create_new(Process&, size_t (Process::*callback)(off_t, BAN::ByteSpan) const, TmpFileSystem&, mode_t, uid_t, gid_t);
|
static BAN::ErrorOr<BAN::RefPtr<ProcROProcessInode>> create_new(Process&, size_t (Process::*callback)(off_t, BAN::ByteSpan) const, TmpFileSystem&, mode_t);
|
||||||
~ProcROInode() = default;
|
~ProcROProcessInode() = default;
|
||||||
|
|
||||||
virtual uid_t uid() const override { return m_process.credentials().ruid(); }
|
virtual uid_t uid() const override { return m_process.credentials().ruid(); }
|
||||||
virtual gid_t gid() const override { return m_process.credentials().rgid(); }
|
virtual gid_t gid() const override { return m_process.credentials().rgid(); }
|
||||||
|
@ -49,11 +49,35 @@ namespace Kernel
|
||||||
virtual bool has_error_impl() const override { return false; }
|
virtual bool has_error_impl() const override { return false; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
ProcROInode(Process&, size_t (Process::*)(off_t, BAN::ByteSpan) const, TmpFileSystem&, const TmpInodeInfo&);
|
ProcROProcessInode(Process&, size_t (Process::*)(off_t, BAN::ByteSpan) const, TmpFileSystem&, const TmpInodeInfo&);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Process& m_process;
|
Process& m_process;
|
||||||
size_t (Process::*m_callback)(off_t, BAN::ByteSpan) const;
|
size_t (Process::*m_callback)(off_t, BAN::ByteSpan) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class ProcROInode final : public TmpInode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
static BAN::ErrorOr<BAN::RefPtr<ProcROInode>> create_new(size_t (*callback)(off_t, BAN::ByteSpan), TmpFileSystem&, mode_t, uid_t, gid_t);
|
||||||
|
~ProcROInode() = default;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
virtual BAN::ErrorOr<size_t> read_impl(off_t, BAN::ByteSpan) override;
|
||||||
|
|
||||||
|
// You may not write here and this is always non blocking
|
||||||
|
virtual BAN::ErrorOr<size_t> write_impl(off_t, BAN::ConstByteSpan) override { return BAN::Error::from_errno(EINVAL); }
|
||||||
|
virtual BAN::ErrorOr<void> truncate_impl(size_t) override { return BAN::Error::from_errno(EINVAL); }
|
||||||
|
|
||||||
|
virtual bool can_read_impl() const override { return true; }
|
||||||
|
virtual bool can_write_impl() const override { return false; }
|
||||||
|
virtual bool has_error_impl() const override { return false; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
ProcROInode(size_t (*callback)(off_t, BAN::ByteSpan), TmpFileSystem&, const TmpInodeInfo&);
|
||||||
|
|
||||||
|
private:
|
||||||
|
size_t (*m_callback)(off_t, BAN::ByteSpan);
|
||||||
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -76,7 +76,7 @@ namespace Kernel
|
||||||
|
|
||||||
bool is_userspace() const { return m_is_userspace; }
|
bool is_userspace() const { return m_is_userspace; }
|
||||||
|
|
||||||
size_t virtual_page_count() const { return (m_kernel_stack->size() / PAGE_SIZE) + (m_userspace_stack->size() / PAGE_SIZE); }
|
size_t virtual_page_count() const { return (m_kernel_stack ? (m_kernel_stack->size() / PAGE_SIZE) : 0) + (m_userspace_stack ? (m_userspace_stack->size() / PAGE_SIZE) : 0); }
|
||||||
size_t physical_page_count() const { return virtual_page_count(); }
|
size_t physical_page_count() const { return virtual_page_count(); }
|
||||||
|
|
||||||
InterruptStack& interrupt_stack() { return m_interrupt_stack; }
|
InterruptStack& interrupt_stack() { return m_interrupt_stack; }
|
||||||
|
|
|
@ -14,6 +14,26 @@ namespace Kernel
|
||||||
ASSERT(s_instance);
|
ASSERT(s_instance);
|
||||||
|
|
||||||
MUST(s_instance->TmpFileSystem::initialize(0555, 0, 0));
|
MUST(s_instance->TmpFileSystem::initialize(0555, 0, 0));
|
||||||
|
|
||||||
|
auto meminfo_inode = MUST(ProcROInode::create_new(
|
||||||
|
[](off_t offset, BAN::ByteSpan buffer) -> size_t
|
||||||
|
{
|
||||||
|
ASSERT(offset >= 0);
|
||||||
|
if ((size_t)offset >= sizeof(full_meminfo_t))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
full_meminfo_t meminfo;
|
||||||
|
meminfo.page_size = PAGE_SIZE;
|
||||||
|
meminfo.free_pages = Heap::get().free_pages();
|
||||||
|
meminfo.used_pages = Heap::get().used_pages();
|
||||||
|
|
||||||
|
size_t bytes = BAN::Math::min<size_t>(sizeof(full_meminfo_t) - offset, buffer.size());
|
||||||
|
memcpy(buffer.data(), (uint8_t*)&meminfo + offset, bytes);
|
||||||
|
return bytes;
|
||||||
|
},
|
||||||
|
*s_instance, 0444, 0, 0
|
||||||
|
));
|
||||||
|
MUST(static_cast<TmpDirectoryInode*>(s_instance->root_inode().ptr())->link_inode(*meminfo_inode, "meminfo"_sv));
|
||||||
}
|
}
|
||||||
|
|
||||||
ProcFileSystem& ProcFileSystem::get()
|
ProcFileSystem& ProcFileSystem::get()
|
||||||
|
@ -30,7 +50,7 @@ namespace Kernel
|
||||||
BAN::ErrorOr<void> ProcFileSystem::on_process_create(Process& process)
|
BAN::ErrorOr<void> ProcFileSystem::on_process_create(Process& process)
|
||||||
{
|
{
|
||||||
auto path = TRY(BAN::String::formatted("{}", process.pid()));
|
auto path = TRY(BAN::String::formatted("{}", process.pid()));
|
||||||
auto inode = TRY(ProcPidInode::create_new(process, *this, 0555, process.credentials().ruid(), process.credentials().rgid()));
|
auto inode = TRY(ProcPidInode::create_new(process, *this, 0555));
|
||||||
TRY(static_cast<TmpDirectoryInode*>(root_inode().ptr())->link_inode(*inode, path));
|
TRY(static_cast<TmpDirectoryInode*>(root_inode().ptr())->link_inode(*inode, path));
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,18 +3,18 @@
|
||||||
namespace Kernel
|
namespace Kernel
|
||||||
{
|
{
|
||||||
|
|
||||||
BAN::ErrorOr<BAN::RefPtr<ProcPidInode>> ProcPidInode::create_new(Process& process, TmpFileSystem& fs, mode_t mode, uid_t uid, gid_t gid)
|
BAN::ErrorOr<BAN::RefPtr<ProcPidInode>> ProcPidInode::create_new(Process& process, TmpFileSystem& fs, mode_t mode)
|
||||||
{
|
{
|
||||||
auto inode_info = create_inode_info(Mode::IFDIR | mode, uid, gid);
|
auto inode_info = create_inode_info(Mode::IFDIR | mode, 0, 0);
|
||||||
|
|
||||||
auto* inode_ptr = new ProcPidInode(process, fs, inode_info);
|
auto* inode_ptr = new ProcPidInode(process, fs, inode_info);
|
||||||
if (inode_ptr == nullptr)
|
if (inode_ptr == nullptr)
|
||||||
return BAN::Error::from_errno(ENOMEM);
|
return BAN::Error::from_errno(ENOMEM);
|
||||||
auto inode = BAN::RefPtr<ProcPidInode>::adopt(inode_ptr);
|
auto inode = BAN::RefPtr<ProcPidInode>::adopt(inode_ptr);
|
||||||
|
|
||||||
TRY(inode->link_inode(*MUST(ProcROInode::create_new(process, &Process::proc_meminfo, fs, 0400, uid, gid)), "meminfo"_sv));
|
TRY(inode->link_inode(*MUST(ProcROProcessInode::create_new(process, &Process::proc_meminfo, fs, 0400)), "meminfo"_sv));
|
||||||
TRY(inode->link_inode(*MUST(ProcROInode::create_new(process, &Process::proc_cmdline, fs, 0400, uid, gid)), "cmdline"_sv));
|
TRY(inode->link_inode(*MUST(ProcROProcessInode::create_new(process, &Process::proc_cmdline, fs, 0400)), "cmdline"_sv));
|
||||||
TRY(inode->link_inode(*MUST(ProcROInode::create_new(process, &Process::proc_environ, fs, 0400, uid, gid)), "environ"_sv));
|
TRY(inode->link_inode(*MUST(ProcROProcessInode::create_new(process, &Process::proc_environ, fs, 0400)), "environ"_sv));
|
||||||
|
|
||||||
return inode;
|
return inode;
|
||||||
}
|
}
|
||||||
|
@ -32,17 +32,17 @@ namespace Kernel
|
||||||
(void)TmpDirectoryInode::unlink_impl("environ"_sv);
|
(void)TmpDirectoryInode::unlink_impl("environ"_sv);
|
||||||
}
|
}
|
||||||
|
|
||||||
BAN::ErrorOr<BAN::RefPtr<ProcROInode>> ProcROInode::create_new(Process& process, size_t (Process::*callback)(off_t, BAN::ByteSpan) const, TmpFileSystem& fs, mode_t mode, uid_t uid, gid_t gid)
|
BAN::ErrorOr<BAN::RefPtr<ProcROProcessInode>> ProcROProcessInode::create_new(Process& process, size_t (Process::*callback)(off_t, BAN::ByteSpan) const, TmpFileSystem& fs, mode_t mode)
|
||||||
{
|
{
|
||||||
auto inode_info = create_inode_info(Mode::IFREG | mode, uid, gid);
|
auto inode_info = create_inode_info(Mode::IFREG | mode, 0, 0);
|
||||||
|
|
||||||
auto* inode_ptr = new ProcROInode(process, callback, fs, inode_info);
|
auto* inode_ptr = new ProcROProcessInode(process, callback, fs, inode_info);
|
||||||
if (inode_ptr == nullptr)
|
if (inode_ptr == nullptr)
|
||||||
return BAN::Error::from_errno(ENOMEM);
|
return BAN::Error::from_errno(ENOMEM);
|
||||||
return BAN::RefPtr<ProcROInode>::adopt(inode_ptr);
|
return BAN::RefPtr<ProcROProcessInode>::adopt(inode_ptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
ProcROInode::ProcROInode(Process& process, size_t (Process::*callback)(off_t, BAN::ByteSpan) const, TmpFileSystem& fs, const TmpInodeInfo& inode_info)
|
ProcROProcessInode::ProcROProcessInode(Process& process, size_t (Process::*callback)(off_t, BAN::ByteSpan) const, TmpFileSystem& fs, const TmpInodeInfo& inode_info)
|
||||||
: TmpInode(fs, MUST(fs.allocate_inode(inode_info)), inode_info)
|
: TmpInode(fs, MUST(fs.allocate_inode(inode_info)), inode_info)
|
||||||
, m_process(process)
|
, m_process(process)
|
||||||
, m_callback(callback)
|
, m_callback(callback)
|
||||||
|
@ -50,12 +50,35 @@ namespace Kernel
|
||||||
m_inode_info.mode |= Inode::Mode::IFREG;
|
m_inode_info.mode |= Inode::Mode::IFREG;
|
||||||
}
|
}
|
||||||
|
|
||||||
BAN::ErrorOr<size_t> ProcROInode::read_impl(off_t offset, BAN::ByteSpan buffer)
|
BAN::ErrorOr<size_t> ProcROProcessInode::read_impl(off_t offset, BAN::ByteSpan buffer)
|
||||||
{
|
{
|
||||||
ASSERT(offset >= 0);
|
if (offset < 0)
|
||||||
if ((size_t)offset >= sizeof(proc_meminfo_t))
|
return BAN::Error::from_errno(EINVAL);
|
||||||
return 0;
|
|
||||||
return (m_process.*m_callback)(offset, buffer);
|
return (m_process.*m_callback)(offset, buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BAN::ErrorOr<BAN::RefPtr<ProcROInode>> ProcROInode::create_new(size_t (*callback)(off_t, BAN::ByteSpan), TmpFileSystem& fs, mode_t mode, uid_t uid, gid_t gid)
|
||||||
|
{
|
||||||
|
auto inode_info = create_inode_info(Mode::IFREG | mode, uid, gid);
|
||||||
|
|
||||||
|
auto* inode_ptr = new ProcROInode(callback, fs, inode_info);
|
||||||
|
if (inode_ptr == nullptr)
|
||||||
|
return BAN::Error::from_errno(ENOMEM);
|
||||||
|
return BAN::RefPtr<ProcROInode>::adopt(inode_ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
ProcROInode::ProcROInode(size_t (*callback)(off_t, BAN::ByteSpan), TmpFileSystem& fs, const TmpInodeInfo& inode_info)
|
||||||
|
: TmpInode(fs, MUST(fs.allocate_inode(inode_info)), inode_info)
|
||||||
|
, m_callback(callback)
|
||||||
|
{
|
||||||
|
m_inode_info.mode |= Inode::Mode::IFREG;
|
||||||
|
}
|
||||||
|
|
||||||
|
BAN::ErrorOr<size_t> ProcROInode::read_impl(off_t offset, BAN::ByteSpan buffer)
|
||||||
|
{
|
||||||
|
if (offset < 0)
|
||||||
|
return BAN::Error::from_errno(EINVAL);
|
||||||
|
return m_callback(offset, buffer);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,6 +24,13 @@ struct proc_meminfo_t
|
||||||
size_t phys_pages;
|
size_t phys_pages;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct full_meminfo_t
|
||||||
|
{
|
||||||
|
size_t page_size;
|
||||||
|
size_t free_pages;
|
||||||
|
size_t used_pages;
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
fildes: refers to valid tty device
|
fildes: refers to valid tty device
|
||||||
command: one of TTY_CMD_* definitions
|
command: one of TTY_CMD_* definitions
|
||||||
|
|
Loading…
Reference in New Issue