198 lines
3.8 KiB
C++

/*
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/>.
*/
#ifndef __FENNIX_KERNEL_TTY_H__
#define __FENNIX_KERNEL_TTY_H__
#include <termios.h>
#include <mutex>
namespace TTY
{
class TerminalBuffer
{
private:
std::vector<char> Buffer;
size_t ReadIndex;
size_t WriteIndex;
std::mutex Mutex;
public:
TerminalBuffer(size_t Size) : Buffer(Size), ReadIndex(0), WriteIndex(0) {}
ssize_t Read(char *OutputBuffer, size_t Size)
{
std::lock_guard<std::mutex> lock(Mutex);
size_t bytesRead = 0;
while (bytesRead < Size && ReadIndex != WriteIndex)
{
OutputBuffer[bytesRead++] = Buffer[ReadIndex];
ReadIndex = (ReadIndex + 1) % Buffer.size();
}
return bytesRead;
}
ssize_t Write(const char *InputBuffer, size_t Size)
{
std::lock_guard<std::mutex> lock(Mutex);
size_t bytesWritten = 0;
for (size_t i = 0; i < Size; ++i)
{
Buffer[WriteIndex] = InputBuffer[i];
WriteIndex = (WriteIndex + 1) % Buffer.size();
bytesWritten++;
}
return bytesWritten;
}
void DrainOutput()
{
std::lock_guard<std::mutex> lock(Mutex);
ReadIndex = WriteIndex;
}
size_t AvailableToRead() const
{
return (WriteIndex - ReadIndex + Buffer.size()) % Buffer.size();
}
size_t AvailableToWrite() const
{
return Buffer.size() - AvailableToRead() - 1;
}
};
class TeletypeDriver
{
protected:
termios TerminalConfig{};
winsize TerminalSize{};
TerminalBuffer TermBuf;
pid_t ProcessGroup;
public:
virtual int Open(int Flags, mode_t Mode);
virtual int Close();
virtual ssize_t Read(void *Buffer, size_t Size, off_t Offset);
virtual ssize_t Write(const void *Buffer, size_t Size, off_t Offset);
virtual int Ioctl(unsigned long Request, void *Argp);
TeletypeDriver();
virtual ~TeletypeDriver() = default;
};
class PTYDevice
{
private:
class PTYMaster
{
private:
TerminalBuffer TermBuf;
public:
PTYMaster() : TermBuf(1024) {}
~PTYMaster() = default;
ssize_t Read(void *Buffer, size_t Size)
{
return TermBuf.Read((char *)Buffer, Size);
}
ssize_t Write(const void *Buffer, size_t Size)
{
return TermBuf.Write((const char *)Buffer, Size);
}
};
class PTYSlave
{
private:
TerminalBuffer TermBuf;
public:
PTYSlave() : TermBuf(1024) {}
~PTYSlave() = default;
ssize_t Read(void *Buffer, size_t Size)
{
return TermBuf.Read((char *)Buffer, Size);
}
ssize_t Write(const void *Buffer, size_t Size)
{
return TermBuf.Write((const char *)Buffer, Size);
}
};
PTYMaster Master;
PTYSlave Slave;
public:
PTYDevice() : Master(), Slave() {}
~PTYDevice() = default;
int Open()
{
stub;
return -ENOSYS;
}
int Close()
{
stub;
return -ENOSYS;
}
ssize_t Read(void *Buffer, size_t Size)
{
return Slave.Read(Buffer, Size);
}
ssize_t Write(const void *Buffer, size_t Size)
{
return Master.Write(Buffer, Size);
}
};
class PTMXDevice
{
private:
std::vector<PTYDevice *> PTYs;
std::mutex PTYMutex;
public:
PTMXDevice() = default;
~PTMXDevice()
{
for (auto pty : PTYs)
delete pty;
}
int Open();
int Close();
PTYDevice *CreatePTY();
int RemovePTY(PTYDevice *pty);
};
}
#endif // !__FENNIX_KERNEL_TTY_H__