2019-01-26 05:28:02 +01:00
|
|
|
#include "WSMessageLoop.h"
|
|
|
|
#include "WSMessage.h"
|
|
|
|
#include "WSMessageReceiver.h"
|
2019-01-16 16:03:50 +01:00
|
|
|
#include "WSWindowManager.h"
|
|
|
|
#include "WSScreen.h"
|
|
|
|
#include "PS2MouseDevice.h"
|
2019-01-16 17:20:58 +01:00
|
|
|
#include <Kernel/Keyboard.h>
|
|
|
|
#include <AK/Bitmap.h>
|
|
|
|
#include "Process.h"
|
2019-01-16 16:03:50 +01:00
|
|
|
|
|
|
|
//#define WSEVENTLOOP_DEBUG
|
|
|
|
|
2019-01-26 05:28:02 +01:00
|
|
|
static WSMessageLoop* s_the;
|
2019-01-16 16:03:50 +01:00
|
|
|
|
2019-01-26 05:28:02 +01:00
|
|
|
WSMessageLoop::WSMessageLoop()
|
2019-02-07 11:12:23 +01:00
|
|
|
: m_lock("WSMessageLoop")
|
2019-01-16 16:03:50 +01:00
|
|
|
{
|
|
|
|
if (!s_the)
|
|
|
|
s_the = this;
|
|
|
|
}
|
|
|
|
|
2019-01-26 05:28:02 +01:00
|
|
|
WSMessageLoop::~WSMessageLoop()
|
2019-01-16 16:03:50 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2019-01-26 05:28:02 +01:00
|
|
|
WSMessageLoop& WSMessageLoop::the()
|
2019-01-16 16:03:50 +01:00
|
|
|
{
|
|
|
|
ASSERT(s_the);
|
|
|
|
return *s_the;
|
|
|
|
}
|
|
|
|
|
2019-01-26 05:28:02 +01:00
|
|
|
int WSMessageLoop::exec()
|
2019-01-16 16:03:50 +01:00
|
|
|
{
|
2019-02-08 09:47:57 +01:00
|
|
|
ASSERT(m_server_process == current);
|
2019-01-16 17:20:58 +01:00
|
|
|
|
|
|
|
m_keyboard_fd = m_server_process->sys$open("/dev/keyboard", O_RDONLY);
|
|
|
|
m_mouse_fd = m_server_process->sys$open("/dev/psaux", O_RDONLY);
|
|
|
|
|
|
|
|
ASSERT(m_keyboard_fd >= 0);
|
|
|
|
ASSERT(m_mouse_fd >= 0);
|
|
|
|
|
2019-01-16 16:03:50 +01:00
|
|
|
m_running = true;
|
|
|
|
for (;;) {
|
2019-01-26 05:35:45 +01:00
|
|
|
wait_for_message();
|
2019-01-16 17:20:58 +01:00
|
|
|
|
2019-01-26 05:35:45 +01:00
|
|
|
Vector<QueuedMessage> messages;
|
2019-01-16 16:03:50 +01:00
|
|
|
{
|
2019-01-17 16:21:04 +01:00
|
|
|
ASSERT_INTERRUPTS_ENABLED();
|
2019-01-16 16:03:50 +01:00
|
|
|
LOCKER(m_lock);
|
2019-01-26 05:35:45 +01:00
|
|
|
messages = move(m_queued_messages);
|
2019-01-16 16:03:50 +01:00
|
|
|
}
|
|
|
|
|
2019-01-26 05:35:45 +01:00
|
|
|
for (auto& queued_message : messages) {
|
|
|
|
auto* receiver = queued_message.receiver;
|
|
|
|
auto& message = *queued_message.message;
|
2019-01-16 16:03:50 +01:00
|
|
|
#ifdef WSEVENTLOOP_DEBUG
|
2019-01-26 05:35:45 +01:00
|
|
|
dbgprintf("WSMessageLoop: receiver{%p} message %u (%s)\n", receiver, (unsigned)event.type(), event.name());
|
2019-01-16 16:03:50 +01:00
|
|
|
#endif
|
|
|
|
if (!receiver) {
|
2019-01-26 05:35:45 +01:00
|
|
|
dbgprintf("WSMessage type %u with no receiver :(\n", message.type());
|
2019-01-16 16:03:50 +01:00
|
|
|
ASSERT_NOT_REACHED();
|
|
|
|
return 1;
|
|
|
|
} else {
|
2019-01-26 05:35:45 +01:00
|
|
|
receiver->on_message(message);
|
2019-01-16 16:03:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-11 13:05:51 +01:00
|
|
|
void WSMessageLoop::post_message(WSMessageReceiver* receiver, OwnPtr<WSMessage>&& message)
|
2019-01-16 16:03:50 +01:00
|
|
|
{
|
|
|
|
LOCKER(m_lock);
|
|
|
|
#ifdef WSEVENTLOOP_DEBUG
|
2019-01-26 05:35:45 +01:00
|
|
|
dbgprintf("WSMessageLoop::post_message: {%u} << receiver=%p, message=%p\n", m_queued_messages.size(), receiver, message.ptr());
|
2019-01-16 16:03:50 +01:00
|
|
|
#endif
|
2019-01-17 02:42:52 +01:00
|
|
|
|
2019-01-26 05:45:47 +01:00
|
|
|
if (message->type() == WSMessage::WM_ClientFinishedPaint) {
|
|
|
|
auto& invalidation_message = static_cast<WSClientFinishedPaintMessage&>(*message);
|
2019-01-26 05:35:45 +01:00
|
|
|
for (auto& queued_message : m_queued_messages) {
|
2019-01-26 05:45:47 +01:00
|
|
|
if (receiver == queued_message.receiver && queued_message.message->type() == WSMessage::WM_ClientFinishedPaint) {
|
|
|
|
auto& queued_invalidation_message = static_cast<WSClientFinishedPaintMessage&>(*queued_message.message);
|
2019-01-26 05:35:45 +01:00
|
|
|
if (queued_invalidation_message.rect().is_empty() || queued_invalidation_message.rect().contains(invalidation_message.rect())) {
|
2019-01-17 02:42:52 +01:00
|
|
|
#ifdef WSEVENTLOOP_DEBUG
|
2019-01-26 05:45:47 +01:00
|
|
|
dbgprintf("Swallow WM_ClientFinishedPaint\n");
|
2019-01-17 02:42:52 +01:00
|
|
|
#endif
|
2019-01-21 07:28:04 +01:00
|
|
|
return;
|
|
|
|
}
|
2019-01-17 02:42:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-26 05:45:47 +01:00
|
|
|
if (message->type() == WSMessage::WM_ClientWantsToPaint) {
|
|
|
|
auto& invalidation_message = static_cast<WSClientWantsToPaintMessage&>(*message);
|
2019-01-26 05:35:45 +01:00
|
|
|
for (auto& queued_message : m_queued_messages) {
|
2019-01-26 05:45:47 +01:00
|
|
|
if (receiver == queued_message.receiver && queued_message.message->type() == WSMessage::WM_ClientWantsToPaint) {
|
|
|
|
auto& queued_invalidation_message = static_cast<WSClientWantsToPaintMessage&>(*queued_message.message);
|
2019-01-26 05:35:45 +01:00
|
|
|
if (queued_invalidation_message.rect().is_empty() || queued_invalidation_message.rect().contains(invalidation_message.rect())) {
|
2019-01-26 05:20:32 +01:00
|
|
|
#ifdef WSEVENTLOOP_DEBUG
|
2019-01-26 05:45:47 +01:00
|
|
|
dbgprintf("Swallow WM_ClientWantsToPaint\n");
|
2019-01-26 05:20:32 +01:00
|
|
|
#endif
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-26 05:35:45 +01:00
|
|
|
m_queued_messages.append({ receiver, move(message) });
|
2019-01-16 17:20:58 +01:00
|
|
|
|
|
|
|
if (current != m_server_process)
|
|
|
|
m_server_process->request_wakeup();
|
2019-01-16 16:03:50 +01:00
|
|
|
}
|
|
|
|
|
2019-01-26 05:35:45 +01:00
|
|
|
void WSMessageLoop::wait_for_message()
|
2019-01-16 17:20:58 +01:00
|
|
|
{
|
|
|
|
fd_set rfds;
|
|
|
|
memset(&rfds, 0, sizeof(rfds));
|
|
|
|
auto bitmap = Bitmap::wrap((byte*)&rfds, FD_SETSIZE);
|
|
|
|
bitmap.set(m_keyboard_fd, true);
|
|
|
|
bitmap.set(m_mouse_fd, true);
|
|
|
|
Syscall::SC_select_params params;
|
|
|
|
params.nfds = max(m_keyboard_fd, m_mouse_fd) + 1;
|
|
|
|
params.readfds = &rfds;
|
|
|
|
params.writefds = nullptr;
|
|
|
|
params.exceptfds = nullptr;
|
2019-01-18 05:26:45 +01:00
|
|
|
struct timeval timeout = { 0, 0 };
|
2019-01-26 05:35:45 +01:00
|
|
|
if (m_queued_messages.is_empty())
|
2019-01-18 05:26:45 +01:00
|
|
|
params.timeout = nullptr;
|
|
|
|
else
|
|
|
|
params.timeout = &timeout;
|
2019-01-16 17:20:58 +01:00
|
|
|
int rc = m_server_process->sys$select(¶ms);
|
|
|
|
memory_barrier();
|
|
|
|
if (rc < 0) {
|
|
|
|
ASSERT_NOT_REACHED();
|
|
|
|
}
|
|
|
|
|
2019-01-17 02:34:08 +01:00
|
|
|
if (bitmap.get(m_keyboard_fd))
|
2019-01-16 17:20:58 +01:00
|
|
|
drain_keyboard();
|
2019-01-17 02:34:08 +01:00
|
|
|
if (bitmap.get(m_mouse_fd))
|
2019-01-16 17:20:58 +01:00
|
|
|
drain_mouse();
|
|
|
|
}
|
|
|
|
|
2019-01-26 05:28:02 +01:00
|
|
|
void WSMessageLoop::drain_mouse()
|
2019-01-16 16:03:50 +01:00
|
|
|
{
|
|
|
|
auto& screen = WSScreen::the();
|
2019-01-16 17:20:58 +01:00
|
|
|
auto& mouse = PS2MouseDevice::the();
|
2019-01-16 16:03:50 +01:00
|
|
|
bool prev_left_button = screen.left_mouse_button_pressed();
|
|
|
|
bool prev_right_button = screen.right_mouse_button_pressed();
|
|
|
|
int dx = 0;
|
|
|
|
int dy = 0;
|
|
|
|
while (mouse.can_read(*m_server_process)) {
|
2019-01-17 02:32:40 +01:00
|
|
|
byte data[3];
|
2019-01-16 17:20:58 +01:00
|
|
|
ssize_t nread = mouse.read(*m_server_process, (byte*)data, sizeof(data));
|
|
|
|
ASSERT(nread == sizeof(data));
|
2019-01-16 16:03:50 +01:00
|
|
|
bool left_button = data[0] & 1;
|
|
|
|
bool right_button = data[0] & 2;
|
2019-02-07 08:07:37 +01:00
|
|
|
bool x_overflow = data[0] & 0x40;
|
|
|
|
bool y_overflow = data[0] & 0x80;
|
|
|
|
bool x_sign = data[0] & 0x10;
|
|
|
|
bool y_sign = data[0] & 0x20;
|
|
|
|
|
|
|
|
if (x_overflow || y_overflow)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
int x = data[1];
|
|
|
|
int y = data[2];
|
|
|
|
if (x && x_sign)
|
|
|
|
x -= 0x100;
|
|
|
|
if (y && y_sign)
|
|
|
|
y -= 0x100;
|
|
|
|
|
|
|
|
dx += x;
|
|
|
|
dy += -y;
|
2019-01-16 16:03:50 +01:00
|
|
|
if (left_button != prev_left_button || right_button != prev_right_button || !mouse.can_read(*m_server_process)) {
|
|
|
|
prev_left_button = left_button;
|
|
|
|
prev_right_button = right_button;
|
|
|
|
screen.on_receive_mouse_data(dx, dy, left_button, right_button);
|
|
|
|
dx = 0;
|
|
|
|
dy = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-01-16 17:20:58 +01:00
|
|
|
|
2019-01-26 05:28:02 +01:00
|
|
|
void WSMessageLoop::drain_keyboard()
|
2019-01-16 17:20:58 +01:00
|
|
|
{
|
|
|
|
auto& screen = WSScreen::the();
|
|
|
|
auto& keyboard = Keyboard::the();
|
|
|
|
while (keyboard.can_read(*m_server_process)) {
|
2019-01-21 07:05:31 +01:00
|
|
|
Keyboard::Event event;
|
|
|
|
ssize_t nread = keyboard.read(*m_server_process, (byte*)&event, sizeof(Keyboard::Event));
|
|
|
|
ASSERT(nread == sizeof(Keyboard::Event));
|
|
|
|
screen.on_receive_keyboard_data(event);
|
2019-01-16 17:20:58 +01:00
|
|
|
}
|
|
|
|
}
|