mirror of
https://github.com/EnderIce2/Fennix.git
synced 2025-07-01 02:19:15 +00:00
feat(kernel/vfs): implement RAMFS filesystem
This commit is contained in:
149
Kernel/include/filesystem/ramfs.hpp
Normal file
149
Kernel/include/filesystem/ramfs.hpp
Normal file
@ -0,0 +1,149 @@
|
||||
/*
|
||||
This file is part of Fennix Kernel.
|
||||
|
||||
Fennix Kernel is free software: you can redistribute it and/or
|
||||
modify it under the terms of the GNU General Public License as
|
||||
published by the Free Software Foundation, either version 3 of
|
||||
the License, or (at your option) any later version.
|
||||
|
||||
Fennix Kernel is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with Fennix Kernel. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <filesystem.hpp>
|
||||
#include <memory.hpp>
|
||||
|
||||
namespace vfs
|
||||
{
|
||||
class RAMFS
|
||||
{
|
||||
public:
|
||||
class InodeBuffer
|
||||
{
|
||||
public:
|
||||
void *Data = nullptr;
|
||||
size_t DataSize = 0;
|
||||
|
||||
void Allocate(size_t size, bool extend = false, bool atEnd = true)
|
||||
{
|
||||
if (extend == false)
|
||||
{
|
||||
if (Data)
|
||||
Free();
|
||||
Data = kmalloc(size);
|
||||
if (!Data)
|
||||
throw std::bad_alloc();
|
||||
DataSize = size;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (Data == nullptr)
|
||||
{
|
||||
Data = kmalloc(size);
|
||||
if (!Data)
|
||||
throw std::bad_alloc();
|
||||
DataSize = size;
|
||||
}
|
||||
else
|
||||
{
|
||||
size_t newSize = DataSize + size;
|
||||
void *newData = kmalloc(newSize);
|
||||
if (!newData)
|
||||
throw std::bad_alloc();
|
||||
|
||||
if (atEnd)
|
||||
memcpy(newData, Data, DataSize);
|
||||
else
|
||||
memcpy(static_cast<char *>(newData) + size, Data, DataSize);
|
||||
|
||||
kfree(Data);
|
||||
Data = newData;
|
||||
DataSize = newSize;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void Free()
|
||||
{
|
||||
if (Data)
|
||||
{
|
||||
kfree(Data);
|
||||
Data = nullptr;
|
||||
DataSize = 0;
|
||||
}
|
||||
}
|
||||
|
||||
bool IsAllocated() const
|
||||
{
|
||||
return Data != nullptr;
|
||||
}
|
||||
|
||||
InodeBuffer() = default;
|
||||
~InodeBuffer() { Free(); }
|
||||
};
|
||||
|
||||
class RAMFSInode
|
||||
{
|
||||
public:
|
||||
struct Inode Node;
|
||||
RAMFSInode *Parent = nullptr;
|
||||
std::string Name;
|
||||
kstat Stat{};
|
||||
mode_t Mode = 0;
|
||||
InodeBuffer Buffer;
|
||||
std::string SymLink;
|
||||
std::vector<RAMFSInode *> Children;
|
||||
|
||||
void AddChild(RAMFSInode *child)
|
||||
{
|
||||
Children.push_back(child);
|
||||
child->Parent = this;
|
||||
}
|
||||
|
||||
void RemoveChild(RAMFSInode *child)
|
||||
{
|
||||
auto it = std::find(Children.begin(), Children.end(), child);
|
||||
if (it != Children.end())
|
||||
{
|
||||
Children.erase(it);
|
||||
child->Parent = nullptr;
|
||||
}
|
||||
}
|
||||
|
||||
RAMFSInode() = default;
|
||||
~RAMFSInode()
|
||||
{
|
||||
for (auto child : Children)
|
||||
delete child;
|
||||
}
|
||||
};
|
||||
|
||||
private:
|
||||
std::unordered_map<ino_t, RAMFSInode *> Files;
|
||||
|
||||
public:
|
||||
dev_t DeviceID = -1;
|
||||
ino_t NextInode = 0;
|
||||
|
||||
int Lookup(struct Inode *Parent, const char *Name, struct Inode **Result);
|
||||
int Create(struct Inode *Parent, const char *Name, mode_t Mode, struct Inode **Result);
|
||||
ssize_t Read(struct Inode *Node, void *Buffer, size_t Size, off_t Offset);
|
||||
ssize_t Write(struct Inode *Node, const void *Buffer, size_t Size, off_t Offset);
|
||||
ssize_t ReadDir(struct Inode *Node, struct kdirent *Buffer, size_t Size, off_t Offset, off_t Entries);
|
||||
int SymLink(struct Inode *Node, const char *Name, const char *Target, struct Inode **Result);
|
||||
ssize_t ReadLink(struct Inode *Node, char *Buffer, size_t Size);
|
||||
int Stat(struct Inode *Node, struct kstat *Stat);
|
||||
|
||||
RAMFS() = default;
|
||||
~RAMFS() = default;
|
||||
};
|
||||
}
|
||||
|
||||
bool MountRAMFS(Inode *Parent, const char *Name, size_t Index)
|
Reference in New Issue
Block a user