#include "stdafx.h" #include "Emu/system_config.h" #include "sys_tty.h" #include #include LOG_CHANNEL(sys_tty); extern fs::file g_tty; extern atomic_t g_tty_size; extern std::array, 16> g_tty_input; extern std::mutex g_tty_mutex; error_code sys_tty_read(s32 ch, vm::ptr buf, u32 len, vm::ptr preadlen) { sys_tty.trace("sys_tty_read(ch=%d, buf=*0x%x, len=%d, preadlen=*0x%x)", ch, buf, len, preadlen); if (!g_cfg.core.debug_console_mode) { return CELL_EIO; } if (ch > 15 || ch < 0 || !buf) { return CELL_EINVAL; } if (ch < SYS_TTYP_USER1) { sys_tty.warning("sys_tty_read called with system channel %d", ch); } usz 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 lock(g_tty_mutex); if (!g_tty_input[ch].empty()) { // reference to our first queue element std::string& input = g_tty_input[ch].front(); // we have to stop reading at either a new line, the param len, or our input string size usz new_line_pos = input.find_first_of('\n'); if (new_line_pos != input.npos) { chars_to_read = std::min(new_line_pos, static_cast(len)); } else { chars_to_read = std::min(input.size(), static_cast(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.empty()) { // pop the first queue element if it was completely consumed g_tty_input[ch].pop_front(); } } } if (!preadlen) { return CELL_EFAULT; } *preadlen = static_cast(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 buf, u32 len, vm::ptr pwritelen) { sys_tty.notice("sys_tty_write(ch=%d, buf=*0x%x, len=%d, pwritelen=*0x%x)", ch, buf, len, pwritelen); std::string msg; if (static_cast(len) > 0 && vm::check_addr(buf.addr(), vm::page_readable, len)) { msg.resize(len); if (!vm::try_access(buf.addr(), msg.data(), len, false)) { msg.clear(); } } // Hack: write to tty even on CEX mode, but disable all error checks if (ch < 0 || ch > 15) { if (g_cfg.core.debug_console_mode) { return CELL_EINVAL; } else { msg.clear(); } } if (g_cfg.core.debug_console_mode) { // Don't modify it in CEX mode len = static_cast(len) > 0 ? len : 0; } if (static_cast(len) > 0) { if (!msg.empty()) { sys_tty.notice(u8"sys_tty_write(): ā€œ%sā€", msg); if (g_tty) { // Lock size by making it negative g_tty_size -= (1ll << 48); g_tty.write(msg); g_tty_size += (1ll << 48) + len; } } else if (g_cfg.core.debug_console_mode) { return {CELL_EFAULT, buf.addr()}; } } if (!pwritelen.try_write(len)) { return {CELL_EFAULT, pwritelen}; } return CELL_OK; }