rpcsx/rpcs3/Emu/Cell/lv2/sys_tty.cpp

115 lines
2.5 KiB
C++
Raw Normal View History

#include "stdafx.h"
#include "sys_tty.h"
2018-06-29 07:59:56 +02:00
#include <deque>
#include <mutex>
LOG_CHANNEL(sys_tty);
2014-08-23 16:51:51 +02:00
extern fs::file g_tty;
extern atomic_t<s64> g_tty_size;
2018-06-29 07:59:56 +02:00
extern std::array<std::deque<std::string>, 16> g_tty_input;
extern std::mutex g_tty_mutex;
error_code sys_tty_read(s32 ch, vm::ptr<char> buf, u32 len, vm::ptr<u32> preadlen)
{
2018-06-29 07:59:56 +02:00
sys_tty.trace("sys_tty_read(ch=%d, buf=*0x%x, len=%d, preadlen=*0x%x)", ch, buf, len, preadlen);
if (false) // TODO: debug mode check
{
return CELL_EIO;
}
if (ch > 15 || !buf)
{
return CELL_EINVAL;
}
if (ch < SYS_TTYP_USER1)
{
sys_tty.warning("sys_tty_read called with system channel %d", ch);
}
size_t chars_to_read = 0; // number of chars that will be read from the input string
std::string tty_read; // string for storage of read chars
if (len > 0)
{
std::lock_guard<std::mutex> lock(g_tty_mutex);
if (g_tty_input[ch].size() > 0)
{
// reference to our first queue element
std::string& input = g_tty_input[ch].front();
2014-08-23 16:51:51 +02:00
2018-06-29 07:59:56 +02:00
// we have to stop reading at either a new line, the param len, or our input string size
size_t new_line_pos = input.find_first_of("\n");
if (new_line_pos != input.npos)
{
chars_to_read = std::min(new_line_pos, static_cast<size_t>(len));
}
else
{
chars_to_read = std::min(input.size(), static_cast<size_t>(len));
}
// read the previously calculated number of chars from the beginning of the input string
tty_read = input.substr(0, chars_to_read);
// remove the just read text from the input string
input = input.substr(chars_to_read, input.size() - 1);
if (input.size() == 0)
{
// pop the first queue element if it was completely consumed
g_tty_input[ch].pop_front();
}
}
}
if (!preadlen)
{
return CELL_EFAULT;
}
*preadlen = (u32)chars_to_read;
if (chars_to_read > 0)
{
std::memcpy(buf.get_ptr(), tty_read.c_str(), chars_to_read);
sys_tty.success("sys_tty_read(ch=%d, len=%d) read %s with length %d", ch, len, tty_read, *preadlen);
}
return CELL_OK;
}
error_code sys_tty_write(s32 ch, vm::cptr<char> buf, u32 len, vm::ptr<u32> pwritelen)
{
sys_tty.notice("sys_tty_write(ch=%d, buf=*0x%x, len=%d, pwritelen=*0x%x)", ch, buf, len, pwritelen);
2014-08-23 16:51:51 +02:00
2015-07-09 19:19:29 +02:00
if (ch > 15)
{
return CELL_EINVAL;
}
2018-02-09 15:49:37 +01:00
const u32 written_len = static_cast<s32>(len) > 0 ? len : 0;
2015-07-09 19:19:29 +02:00
if (written_len > 0 && g_tty)
2015-07-09 19:19:29 +02:00
{
// Lock size by making it negative
g_tty_size -= (1ll << 48);
g_tty.write(buf.get_ptr(), len);
g_tty_size += (1ll << 48) + len;
2015-07-09 19:19:29 +02:00
}
2018-02-09 15:49:37 +01:00
if (!pwritelen)
{
return CELL_EFAULT;
}
*pwritelen = written_len;
2018-02-09 15:49:37 +01:00
return CELL_OK;
}