123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521 |
- #include <WindowServer/WSClientConnection.h>
- #include <WindowServer/WSMessageLoop.h>
- #include <WindowServer/WSMenuBar.h>
- #include <WindowServer/WSMenu.h>
- #include <WindowServer/WSMenuItem.h>
- #include <WindowServer/WSWindow.h>
- #include <WindowServer/WSWindowManager.h>
- #include <WindowServer/WSAPITypes.h>
- #include <WindowServer/WSClipboard.h>
- #include <SharedBuffer.h>
- #include <sys/ioctl.h>
- #include <unistd.h>
- #include <errno.h>
- #include <stdio.h>
- HashMap<int, WSClientConnection*>* s_connections;
- void WSClientConnection::for_each_client(Function<void(WSClientConnection&)> callback)
- {
- if (!s_connections)
- return;
- for (auto& it : *s_connections) {
- callback(*it.value);
- }
- }
- WSClientConnection* WSClientConnection::from_client_id(int client_id)
- {
- if (!s_connections)
- return nullptr;
- auto it = s_connections->find(client_id);
- if (it == s_connections->end())
- return nullptr;
- return (*it).value;
- }
- WSClientConnection::WSClientConnection(int fd)
- : m_fd(fd)
- {
- static int s_next_client_id = 0;
- m_client_id = ++s_next_client_id;
- int rc = ioctl(m_fd, 413, (int)&m_pid);
- ASSERT(rc == 0);
- if (!s_connections)
- s_connections = new HashMap<int, WSClientConnection*>;
- s_connections->set(m_client_id, this);
- WSAPI_ServerMessage message;
- message.type = WSAPI_ServerMessage::Type::Greeting;
- message.greeting.server_pid = getpid();
- post_message(message);
- }
- WSClientConnection::~WSClientConnection()
- {
- s_connections->remove(m_client_id);
- int rc = close(m_fd);
- ASSERT(rc == 0);
- }
- void WSClientConnection::post_error(const String& error_message)
- {
- dbgprintf("WSClientConnection::post_error: client_id=%d: %s\n", m_client_id, error_message.characters());
- WSAPI_ServerMessage message;
- message.type = WSAPI_ServerMessage::Type::Error;
- ASSERT(error_message.length() < (ssize_t)sizeof(message.text));
- strcpy(message.text, error_message.characters());
- message.text_length = error_message.length();
- post_message(message);
- }
- void WSClientConnection::post_message(const WSAPI_ServerMessage& message)
- {
- int nwritten = write(m_fd, &message, sizeof(message));
- if (nwritten < 0) {
- if (errno == EPIPE) {
- dbgprintf("WSClientConnection::post_message: Disconnected from peer.\n");
- return;
- }
- perror("WSClientConnection::post_message write");
- ASSERT_NOT_REACHED();
- }
- ASSERT(nwritten == sizeof(message));
- }
- RetainPtr<GraphicsBitmap> WSClientConnection::create_shared_bitmap(GraphicsBitmap::Format format, const Size& size)
- {
- auto shared_buffer = SharedBuffer::create(m_pid, size.area() * sizeof(RGBA32));
- ASSERT(shared_buffer);
- return GraphicsBitmap::create_with_shared_buffer(format, *shared_buffer, size);
- }
- void WSClientConnection::on_message(WSMessage& message)
- {
- if (message.is_client_request()) {
- on_request(static_cast<WSAPIClientRequest&>(message));
- return;
- }
- if (message.type() == WSMessage::WM_ClientDisconnected) {
- int client_id = static_cast<WSClientDisconnectedNotification&>(message).client_id();
- dbgprintf("WSClientConnection: Client disconnected: %d\n", client_id);
- delete this;
- return;
- }
- }
- void WSClientConnection::handle_request(WSAPICreateMenubarRequest&)
- {
- int menubar_id = m_next_menubar_id++;
- auto menubar = make<WSMenuBar>(*this, menubar_id);
- m_menubars.set(menubar_id, move(menubar));
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidCreateMenubar;
- response.menu.menubar_id = menubar_id;
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPIDestroyMenubarRequest& request)
- {
- int menubar_id = request.menubar_id();
- auto it = m_menubars.find(menubar_id);
- if (it == m_menubars.end()) {
- post_error("Bad menubar ID");
- return;
- }
- auto& menubar = *(*it).value;
- WSWindowManager::the().close_menubar(menubar);
- m_menubars.remove(it);
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidDestroyMenubar;
- response.menu.menubar_id = menubar_id;
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPICreateMenuRequest& request)
- {
- int menu_id = m_next_menu_id++;
- auto menu = make<WSMenu>(this, menu_id, request.text());
- m_menus.set(menu_id, move(menu));
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidCreateMenu;
- response.menu.menu_id = menu_id;
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPIDestroyMenuRequest& request)
- {
- int menu_id = static_cast<WSAPIDestroyMenuRequest&>(request).menu_id();
- auto it = m_menus.find(menu_id);
- if (it == m_menus.end()) {
- post_error("Bad menu ID");
- return;
- }
- auto& menu = *(*it).value;
- WSWindowManager::the().close_menu(menu);
- m_menus.remove(it);
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidDestroyMenu;
- response.menu.menu_id = menu_id;
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPISetApplicationMenubarRequest& request)
- {
- int menubar_id = request.menubar_id();
- auto it = m_menubars.find(menubar_id);
- if (it == m_menubars.end()) {
- post_error("Bad menubar ID");
- return;
- }
- auto& menubar = *(*it).value;
- m_app_menubar = menubar.make_weak_ptr();
- WSWindowManager::the().notify_client_changed_app_menubar(*this);
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidSetApplicationMenubar;
- response.menu.menubar_id = menubar_id;
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPIAddMenuToMenubarRequest& request)
- {
- int menubar_id = request.menubar_id();
- int menu_id = request.menu_id();
- auto it = m_menubars.find(menubar_id);
- auto jt = m_menus.find(menu_id);
- if (it == m_menubars.end()) {
- post_error("Bad menubar ID");
- return;
- }
- if (jt == m_menus.end()) {
- post_error("Bad menu ID");
- return;
- }
- auto& menubar = *(*it).value;
- auto& menu = *(*jt).value;
- menubar.add_menu(&menu);
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidAddMenuToMenubar;
- response.menu.menubar_id = menubar_id;
- response.menu.menu_id = menu_id;
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPIAddMenuItemRequest& request)
- {
- int menu_id = request.menu_id();
- unsigned identifier = request.identifier();
- auto it = m_menus.find(menu_id);
- if (it == m_menus.end()) {
- post_error("Bad menu ID");
- return;
- }
- auto& menu = *(*it).value;
- menu.add_item(make<WSMenuItem>(identifier, request.text(), request.shortcut_text()));
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidAddMenuItem;
- response.menu.menu_id = menu_id;
- response.menu.identifier = identifier;
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPIAddMenuSeparatorRequest& request)
- {
- int menu_id = request.menu_id();
- auto it = m_menus.find(menu_id);
- if (it == m_menus.end()) {
- post_error("Bad menu ID");
- return;
- }
- auto& menu = *(*it).value;
- menu.add_item(make<WSMenuItem>(WSMenuItem::Separator));
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidAddMenuSeparator;
- response.menu.menu_id = menu_id;
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPISetWindowOpacityRequest& request)
- {
- int window_id = request.window_id();
- auto it = m_windows.find(window_id);
- if (it == m_windows.end()) {
- post_error("Bad window ID");
- return;
- }
- auto& window = *(*it).value;
- window.set_opacity(request.opacity());
- }
- void WSClientConnection::handle_request(WSAPISetWindowTitleRequest& request)
- {
- int window_id = request.window_id();
- auto it = m_windows.find(window_id);
- if (it == m_windows.end()) {
- post_error("Bad window ID");
- return;
- }
- auto& window = *(*it).value;
- window.set_title(request.title());
- }
- void WSClientConnection::handle_request(WSAPIGetWindowTitleRequest& request)
- {
- int window_id = request.window_id();
- auto it = m_windows.find(window_id);
- if (it == m_windows.end()) {
- post_error("Bad window ID");
- return;
- }
- auto& window = *(*it).value;
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidGetWindowTitle;
- response.window_id = window.window_id();
- ASSERT(window.title().length() < (ssize_t)sizeof(response.text));
- strcpy(response.text, window.title().characters());
- response.text_length = window.title().length();
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPISetWindowRectRequest& request)
- {
- int window_id = request.window_id();
- auto it = m_windows.find(window_id);
- if (it == m_windows.end()) {
- post_error("Bad window ID");
- return;
- }
- auto& window = *(*it).value;
- window.set_rect(request.rect());
- }
- void WSClientConnection::handle_request(WSAPIGetWindowRectRequest& request)
- {
- int window_id = request.window_id();
- auto it = m_windows.find(window_id);
- if (it == m_windows.end()) {
- post_error("Bad window ID");
- return;
- }
- auto& window = *(*it).value;
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidGetWindowRect;
- response.window_id = window.window_id();
- response.window.rect = window.rect();
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPISetClipboardContentsRequest& request)
- {
- auto shared_buffer = SharedBuffer::create_from_shared_buffer_id(request.shared_buffer_id());
- if (!shared_buffer) {
- post_error("Bad shared buffer ID");
- return;
- }
- WSClipboard::the().set_data(*shared_buffer, request.size());
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidSetClipboardContents;
- response.clipboard.shared_buffer_id = shared_buffer->shared_buffer_id();
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPIGetClipboardContentsRequest&)
- {
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidGetClipboardContents;
- response.clipboard.shared_buffer_id = -1;
- response.clipboard.contents_size = 0;
- if (WSClipboard::the().size()) {
- // FIXME: Optimize case where an app is copy/pasting within itself.
- // We can just reuse the SharedBuffer then, since it will have the same peer PID.
- // It would be even nicer if a SharedBuffer could have an arbitrary number of clients..
- RetainPtr<SharedBuffer> shared_buffer = SharedBuffer::create(m_pid, WSClipboard::the().size());
- ASSERT(shared_buffer);
- memcpy(shared_buffer->data(), WSClipboard::the().data(), WSClipboard::the().size());
- response.clipboard.shared_buffer_id = shared_buffer->shared_buffer_id();
- response.clipboard.contents_size = WSClipboard::the().size();
- // FIXME: This is a workaround for the fact that SharedBuffers will go away if neither side is retaining them.
- // After we respond to GetClipboardContents, we have to wait for the client to retain the buffer on his side.
- m_last_sent_clipboard_content = move(shared_buffer);
- }
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPICreateWindowRequest& request)
- {
- int window_id = m_next_window_id++;
- auto window = make<WSWindow>(*this, window_id);
- window->set_has_alpha_channel(request.has_alpha_channel());
- window->set_title(request.title());
- window->set_rect(request.rect());
- window->set_opacity(request.opacity());
- window->set_size_increment(request.size_increment());
- window->set_base_size(request.base_size());
- m_windows.set(window_id, move(window));
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidCreateWindow;
- response.window_id = window_id;
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPIDestroyWindowRequest& request)
- {
- int window_id = request.window_id();
- auto it = m_windows.find(window_id);
- if (it == m_windows.end()) {
- post_error("Bad window ID");
- return;
- }
- auto& window = *(*it).value;
- WSWindowManager::the().invalidate(window);
- m_windows.remove(it);
- }
- void WSClientConnection::handle_request(WSAPIInvalidateRectRequest& request)
- {
- int window_id = request.window_id();
- auto it = m_windows.find(window_id);
- if (it == m_windows.end()) {
- post_error("Bad window ID");
- return;
- }
- auto& window = *(*it).value;
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::Paint;
- response.window_id = window_id;
- response.paint.rect = request.rect();
- response.paint.window_size = window.size();
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPIDidFinishPaintingNotification& request)
- {
- int window_id = request.window_id();
- auto it = m_windows.find(window_id);
- if (it == m_windows.end()) {
- post_error("Bad window ID");
- return;
- }
- auto& window = *(*it).value;
- if (!window.has_painted_since_last_resize()) {
- if (window.last_lazy_resize_rect().size() == request.rect().size()) {
- window.set_has_painted_since_last_resize(true);
- WSMessageLoop::the().post_message(window, make<WSResizeEvent>(window.last_lazy_resize_rect(), window.rect()));
- }
- }
- WSWindowManager::the().invalidate(window, request.rect());
- }
- void WSClientConnection::handle_request(WSAPIGetWindowBackingStoreRequest& request)
- {
- int window_id = request.window_id();
- auto it = m_windows.find(window_id);
- if (it == m_windows.end()) {
- post_error("Bad window ID");
- return;
- }
- auto& window = *(*it).value;
- auto* backing_store = window.backing_store();
- WSAPI_ServerMessage response;
- response.type = WSAPI_ServerMessage::Type::DidGetWindowBackingStore;
- response.window_id = window_id;
- response.backing.bpp = sizeof(RGBA32);
- response.backing.pitch = backing_store->pitch();
- response.backing.size = backing_store->size();
- response.backing.has_alpha_channel = backing_store->has_alpha_channel();
- response.backing.shared_buffer_id = backing_store->shared_buffer_id();
- post_message(response);
- }
- void WSClientConnection::handle_request(WSAPISetWindowBackingStoreRequest& request)
- {
- int window_id = request.window_id();
- auto it = m_windows.find(window_id);
- if (it == m_windows.end()) {
- post_error("Bad window ID");
- return;
- }
- auto& window = *(*it).value;
- auto shared_buffer = SharedBuffer::create_from_shared_buffer_id(request.shared_buffer_id());
- if (!shared_buffer)
- return;
- auto backing_store = GraphicsBitmap::create_with_shared_buffer(
- request.has_alpha_channel() ? GraphicsBitmap::Format::RGBA32 : GraphicsBitmap::Format::RGB32,
- *shared_buffer,
- request.size());
- if (!backing_store)
- return;
- window.set_backing_store(move(backing_store));
- window.invalidate();
- }
- void WSClientConnection::handle_request(WSAPISetGlobalCursorTrackingRequest& request)
- {
- int window_id = request.window_id();
- auto it = m_windows.find(window_id);
- if (it == m_windows.end()) {
- post_error("Bad window ID");
- return;
- }
- auto& window = *(*it).value;
- window.set_global_cursor_tracking_enabled(request.value());
- }
- void WSClientConnection::on_request(WSAPIClientRequest& request)
- {
- switch (request.type()) {
- case WSMessage::APICreateMenubarRequest:
- return handle_request(static_cast<WSAPICreateMenubarRequest&>(request));
- case WSMessage::APIDestroyMenubarRequest:
- return handle_request(static_cast<WSAPIDestroyMenubarRequest&>(request));
- case WSMessage::APICreateMenuRequest:
- return handle_request(static_cast<WSAPICreateMenuRequest&>(request));
- case WSMessage::APIDestroyMenuRequest:
- return handle_request(static_cast<WSAPIDestroyMenuRequest&>(request));
- case WSMessage::APISetApplicationMenubarRequest:
- return handle_request(static_cast<WSAPISetApplicationMenubarRequest&>(request));
- case WSMessage::APIAddMenuToMenubarRequest:
- return handle_request(static_cast<WSAPIAddMenuToMenubarRequest&>(request));
- case WSMessage::APIAddMenuItemRequest:
- return handle_request(static_cast<WSAPIAddMenuItemRequest&>(request));
- case WSMessage::APIAddMenuSeparatorRequest:
- return handle_request(static_cast<WSAPIAddMenuSeparatorRequest&>(request));
- case WSMessage::APISetWindowTitleRequest:
- return handle_request(static_cast<WSAPISetWindowTitleRequest&>(request));
- case WSMessage::APIGetWindowTitleRequest:
- return handle_request(static_cast<WSAPIGetWindowTitleRequest&>(request));
- case WSMessage::APISetWindowRectRequest:
- return handle_request(static_cast<WSAPISetWindowRectRequest&>(request));
- case WSMessage::APIGetWindowRectRequest:
- return handle_request(static_cast<WSAPIGetWindowRectRequest&>(request));
- case WSMessage::APISetClipboardContentsRequest:
- return handle_request(static_cast<WSAPISetClipboardContentsRequest&>(request));
- case WSMessage::APIGetClipboardContentsRequest:
- return handle_request(static_cast<WSAPIGetClipboardContentsRequest&>(request));
- case WSMessage::APICreateWindowRequest:
- return handle_request(static_cast<WSAPICreateWindowRequest&>(request));
- case WSMessage::APIDestroyWindowRequest:
- return handle_request(static_cast<WSAPIDestroyWindowRequest&>(request));
- case WSMessage::APIInvalidateRectRequest:
- return handle_request(static_cast<WSAPIInvalidateRectRequest&>(request));
- case WSMessage::APIDidFinishPaintingNotification:
- return handle_request(static_cast<WSAPIDidFinishPaintingNotification&>(request));
- case WSMessage::APIGetWindowBackingStoreRequest:
- return handle_request(static_cast<WSAPIGetWindowBackingStoreRequest&>(request));
- case WSMessage::APISetGlobalCursorTrackingRequest:
- return handle_request(static_cast<WSAPISetGlobalCursorTrackingRequest&>(request));
- case WSMessage::APISetWindowOpacityRequest:
- return handle_request(static_cast<WSAPISetWindowOpacityRequest&>(request));
- case WSMessage::APISetWindowBackingStoreRequest:
- return handle_request(static_cast<WSAPISetWindowBackingStoreRequest&>(request));
- default:
- break;
- }
- }
|