2023-08-14 16:15:29 +00:00
|
|
|
#include "IPC.h"
|
2023-09-16 09:45:51 +00:00
|
|
|
#include "Mouse.h"
|
2023-08-15 10:56:55 +00:00
|
|
|
#include "Screen.h"
|
2023-08-14 18:08:05 +00:00
|
|
|
#include <luna/Alignment.h>
|
2023-08-14 16:15:29 +00:00
|
|
|
#include <luna/String.h>
|
|
|
|
#include <os/File.h>
|
2023-08-29 14:25:30 +00:00
|
|
|
#include <os/SharedMemory.h>
|
2023-11-22 20:29:59 +00:00
|
|
|
#include <sys/mman.h>
|
2023-08-14 18:08:05 +00:00
|
|
|
#include <time.h>
|
2023-08-14 16:15:29 +00:00
|
|
|
|
2023-08-14 18:08:05 +00:00
|
|
|
#define TRY_OR_IPC_ERROR(expr) \
|
|
|
|
({ \
|
|
|
|
auto _expr_rc = (expr); \
|
|
|
|
if (!_expr_rc.has_value()) \
|
|
|
|
{ \
|
|
|
|
os::IPC::send_error(client.conn, _expr_rc.error()); \
|
|
|
|
return {}; \
|
|
|
|
} \
|
|
|
|
_expr_rc.release_value(); \
|
|
|
|
})
|
|
|
|
|
|
|
|
#define READ_MESSAGE(request) \
|
|
|
|
do { \
|
|
|
|
auto rc = client.conn.recv_typed(request); \
|
|
|
|
if (rc.has_error()) \
|
|
|
|
{ \
|
|
|
|
if (rc.error() == EAGAIN) \
|
|
|
|
{ \
|
|
|
|
client.rpc_in_progress = true; \
|
2023-08-15 08:28:11 +00:00
|
|
|
client.rpc_id = decltype(request)::ID; \
|
2023-08-14 18:08:05 +00:00
|
|
|
return {}; \
|
|
|
|
} \
|
2023-08-16 14:48:39 +00:00
|
|
|
if (rc.error() == EINTR) \
|
|
|
|
{ \
|
|
|
|
client.rpc_in_progress = true; \
|
|
|
|
client.rpc_id = decltype(request)::ID; \
|
|
|
|
return {}; \
|
|
|
|
} \
|
2023-08-14 18:08:05 +00:00
|
|
|
else \
|
|
|
|
return rc.release_error(); \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
2023-08-15 09:20:17 +00:00
|
|
|
#define CHECK_WINDOW_ID(request) \
|
|
|
|
do { \
|
|
|
|
if ((usize)request.window >= client.windows.size() || !client.windows[request.window]) \
|
|
|
|
{ \
|
|
|
|
os::eprintln("wind: Window id is invalid!"); \
|
|
|
|
return {}; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
2023-08-14 18:08:05 +00:00
|
|
|
static Result<void> handle_create_window_message(Client& client)
|
|
|
|
{
|
|
|
|
ui::CreateWindowRequest request;
|
|
|
|
READ_MESSAGE(request);
|
|
|
|
|
2023-08-29 14:25:30 +00:00
|
|
|
request.rect = request.rect.normalized();
|
|
|
|
|
2023-10-13 20:33:36 +00:00
|
|
|
auto name = TRY_OR_IPC_ERROR(String::from_cstring("Window"));
|
2023-08-14 16:15:29 +00:00
|
|
|
|
2023-10-13 20:33:36 +00:00
|
|
|
auto shm_path = TRY_OR_IPC_ERROR(String::format("/wind-shm-%d-%lu"_sv, client.conn.fd(), time(NULL)));
|
2023-09-02 12:02:06 +00:00
|
|
|
|
2023-12-27 11:56:40 +00:00
|
|
|
auto* window = new (std::nothrow) Window(request.rect, move(name));
|
2023-08-14 16:15:29 +00:00
|
|
|
if (!window)
|
|
|
|
{
|
|
|
|
os::IPC::send_error(client.conn, ENOMEM);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-10-13 20:33:36 +00:00
|
|
|
auto guard = make_scope_guard([window] {
|
|
|
|
g_windows.remove(window);
|
|
|
|
delete window;
|
|
|
|
});
|
|
|
|
|
2023-08-29 14:25:30 +00:00
|
|
|
window->pixels = (u32*)TRY_OR_IPC_ERROR(
|
2023-12-27 11:56:40 +00:00
|
|
|
os::SharedMemory::create(shm_path.view(), window->surface.height * window->surface.width * 4));
|
2023-08-14 18:08:05 +00:00
|
|
|
|
|
|
|
TRY_OR_IPC_ERROR(client.windows.try_append(window));
|
|
|
|
int id = static_cast<int>(client.windows.size() - 1);
|
2023-08-14 16:15:29 +00:00
|
|
|
|
2023-10-13 20:33:36 +00:00
|
|
|
// No more fallible operations, this operation is guaranteed to succeed now.
|
|
|
|
guard.deactivate();
|
|
|
|
|
2023-08-15 09:20:17 +00:00
|
|
|
window->client = &client;
|
|
|
|
window->id = id;
|
|
|
|
|
2023-08-14 16:15:29 +00:00
|
|
|
ui::CreateWindowResponse response;
|
|
|
|
response.window = id;
|
2023-08-14 18:08:05 +00:00
|
|
|
SET_IPC_STRING(response.shm_path, shm_path.chars());
|
2023-11-22 20:29:59 +00:00
|
|
|
window->shm_path = move(shm_path);
|
2023-08-14 16:15:29 +00:00
|
|
|
os::IPC::send_async(client.conn, response);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-11-22 20:29:59 +00:00
|
|
|
static Result<void> handle_remove_shm_message(Client& client)
|
|
|
|
{
|
|
|
|
ui::RemoveSharedMemoryRequest request;
|
|
|
|
READ_MESSAGE(request);
|
|
|
|
|
|
|
|
CHECK_WINDOW_ID(request);
|
|
|
|
|
|
|
|
shm_unlink(client.windows[request.window]->shm_path.chars());
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-08-14 18:08:05 +00:00
|
|
|
static Result<void> handle_set_window_title_message(Client& client)
|
|
|
|
{
|
|
|
|
ui::SetWindowTitleRequest request;
|
|
|
|
READ_MESSAGE(request);
|
|
|
|
|
|
|
|
auto name = COPY_IPC_STRING(request.title);
|
|
|
|
|
|
|
|
os::println("wind: SetWindowTitle(\"%s\") for window %d", name.chars(), request.window);
|
|
|
|
|
2023-08-15 09:20:17 +00:00
|
|
|
CHECK_WINDOW_ID(request);
|
2023-08-14 18:08:05 +00:00
|
|
|
|
|
|
|
client.windows[request.window]->name = move(name);
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
static Result<void> handle_invalidate_message(Client& client)
|
|
|
|
{
|
|
|
|
ui::InvalidateRequest request;
|
|
|
|
READ_MESSAGE(request);
|
|
|
|
|
2023-08-15 09:20:17 +00:00
|
|
|
CHECK_WINDOW_ID(request);
|
2023-08-14 18:08:05 +00:00
|
|
|
|
|
|
|
client.windows[request.window]->dirty = true;
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-08-15 09:20:17 +00:00
|
|
|
static Result<void> handle_close_window_message(Client& client)
|
|
|
|
{
|
|
|
|
ui::CloseWindowRequest request;
|
|
|
|
READ_MESSAGE(request);
|
|
|
|
|
|
|
|
CHECK_WINDOW_ID(request);
|
|
|
|
|
|
|
|
auto* window = client.windows[request.window];
|
|
|
|
client.windows[request.window] = nullptr;
|
|
|
|
g_windows.remove(window);
|
2023-09-16 09:45:51 +00:00
|
|
|
Mouse::the().window_did_close(window);
|
2023-08-15 09:20:17 +00:00
|
|
|
delete window;
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-08-15 10:56:55 +00:00
|
|
|
static Result<void> handle_get_screen_rect_message(Client& client)
|
|
|
|
{
|
|
|
|
ui::GetScreenRectRequest request;
|
|
|
|
READ_MESSAGE(request); // Kinda pointless, but required.
|
|
|
|
|
|
|
|
ui::GetScreenRectResponse response;
|
|
|
|
response.rect = Screen::the().canvas().rect();
|
|
|
|
os::IPC::send_async(client.conn, response);
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-12-27 11:56:40 +00:00
|
|
|
static Result<void> handle_set_titlebar_rect_message(Client& client)
|
|
|
|
{
|
|
|
|
ui::SetTitlebarRectRequest request;
|
|
|
|
READ_MESSAGE(request);
|
|
|
|
|
|
|
|
request.titlebar_rect = request.titlebar_rect.normalized();
|
|
|
|
|
|
|
|
CHECK_WINDOW_ID(request);
|
|
|
|
|
|
|
|
auto* window = client.windows[request.window];
|
|
|
|
ui::Rect titlebar_rect = window->surface.absolute(request.titlebar_rect);
|
|
|
|
|
|
|
|
if (!window->surface.contains(titlebar_rect))
|
|
|
|
{
|
|
|
|
os::eprintln("wind: SetTitlebarRect: titlebar rect outside window!");
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
window->titlebar = request.titlebar_rect;
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-08-14 16:15:29 +00:00
|
|
|
namespace wind
|
|
|
|
{
|
|
|
|
Result<void> handle_ipc_message(Client& client, u8 id)
|
|
|
|
{
|
|
|
|
client.rpc_in_progress = false;
|
|
|
|
switch (id)
|
|
|
|
{
|
|
|
|
case ui::CREATE_WINDOW_ID: return handle_create_window_message(client);
|
2023-11-22 20:29:59 +00:00
|
|
|
case ui::REMOVE_SHM_ID: return handle_remove_shm_message(client);
|
2023-08-14 18:08:05 +00:00
|
|
|
case ui::SET_WINDOW_TITLE_ID: return handle_set_window_title_message(client);
|
|
|
|
case ui::INVALIDATE_ID: return handle_invalidate_message(client);
|
2023-08-15 09:20:17 +00:00
|
|
|
case ui::CLOSE_WINDOW_ID: return handle_close_window_message(client);
|
2023-08-15 10:56:55 +00:00
|
|
|
case ui::GET_SCREEN_RECT_ID: return handle_get_screen_rect_message(client);
|
2023-12-27 11:56:40 +00:00
|
|
|
case ui::SET_TITLEBAR_RECT_ID: return handle_set_titlebar_rect_message(client);
|
2023-08-14 16:15:29 +00:00
|
|
|
default: os::eprintln("wind: Invalid IPC message from client!"); return err(EINVAL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<void> handle_ipc(Client& client)
|
|
|
|
{
|
|
|
|
if (client.rpc_in_progress) return handle_ipc_message(client, client.rpc_id);
|
|
|
|
|
|
|
|
u8 id;
|
|
|
|
auto rc = client.conn.recv_typed(id);
|
|
|
|
if (rc.has_error())
|
|
|
|
{
|
|
|
|
if (rc.error() == EAGAIN) { return {}; }
|
2023-08-16 14:48:39 +00:00
|
|
|
if (rc.error() == EINTR) { return {}; }
|
2023-08-14 16:15:29 +00:00
|
|
|
else
|
|
|
|
return rc.release_error();
|
|
|
|
}
|
|
|
|
|
|
|
|
return handle_ipc_message(client, id);
|
|
|
|
}
|
|
|
|
}
|