WSEventLoop.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. #include <WindowServer/WSEventLoop.h>
  2. #include <WindowServer/WSEvent.h>
  3. #include <LibCore/CObject.h>
  4. #include <WindowServer/WSWindowManager.h>
  5. #include <WindowServer/WSScreen.h>
  6. #include <WindowServer/WSClientConnection.h>
  7. #include <WindowServer/WSAPITypes.h>
  8. #include <WindowServer/WSCursor.h>
  9. #include <Kernel/KeyCode.h>
  10. #include <Kernel/MousePacket.h>
  11. #include <sys/socket.h>
  12. #include <sys/select.h>
  13. #include <sys/time.h>
  14. #include <time.h>
  15. #include <unistd.h>
  16. #include <fcntl.h>
  17. #include <stdio.h>
  18. #include <errno.h>
  19. //#define WSMESSAGELOOP_DEBUG
  20. WSEventLoop::WSEventLoop()
  21. {
  22. m_keyboard_fd = open("/dev/keyboard", O_RDONLY | O_NONBLOCK | O_CLOEXEC);
  23. m_mouse_fd = open("/dev/psaux", O_RDONLY | O_NONBLOCK | O_CLOEXEC);
  24. unlink("/tmp/wsportal");
  25. m_server_fd = socket(AF_LOCAL, SOCK_STREAM | SOCK_NONBLOCK | SOCK_CLOEXEC, 0);
  26. ASSERT(m_server_fd >= 0);
  27. sockaddr_un address;
  28. address.sun_family = AF_LOCAL;
  29. strcpy(address.sun_path, "/tmp/wsportal");
  30. int rc = bind(m_server_fd, (const sockaddr*)&address, sizeof(address));
  31. ASSERT(rc == 0);
  32. rc = listen(m_server_fd, 5);
  33. ASSERT(rc == 0);
  34. ASSERT(m_keyboard_fd >= 0);
  35. ASSERT(m_mouse_fd >= 0);
  36. }
  37. WSEventLoop::~WSEventLoop()
  38. {
  39. }
  40. void WSEventLoop::drain_server()
  41. {
  42. sockaddr_un address;
  43. socklen_t address_size = sizeof(address);
  44. int client_fd = accept(m_server_fd, (sockaddr*)&address, &address_size);
  45. if (client_fd < 0) {
  46. dbgprintf("WindowServer: accept() failed: %s\n", strerror(errno));
  47. } else {
  48. new WSClientConnection(client_fd);
  49. }
  50. }
  51. void WSEventLoop::drain_mouse()
  52. {
  53. auto& screen = WSScreen::the();
  54. unsigned prev_buttons = screen.mouse_button_state();
  55. int dx = 0;
  56. int dy = 0;
  57. int dz = 0;
  58. unsigned buttons = prev_buttons;
  59. for (;;) {
  60. MousePacket packet;
  61. ssize_t nread = read(m_mouse_fd, &packet, sizeof(MousePacket));
  62. if (nread == 0)
  63. break;
  64. ASSERT(nread == sizeof(packet));
  65. buttons = packet.buttons;
  66. dx += packet.dx;
  67. dy += -packet.dy;
  68. dz += packet.dz;
  69. if (buttons != prev_buttons) {
  70. screen.on_receive_mouse_data(dx, dy, dz, buttons);
  71. dx = 0;
  72. dy = 0;
  73. dz = 0;
  74. prev_buttons = buttons;
  75. }
  76. }
  77. if (dx || dy || dz)
  78. screen.on_receive_mouse_data(dx, dy, dz, buttons);
  79. }
  80. void WSEventLoop::drain_keyboard()
  81. {
  82. auto& screen = WSScreen::the();
  83. for (;;) {
  84. KeyEvent event;
  85. ssize_t nread = read(m_keyboard_fd, (byte*)&event, sizeof(KeyEvent));
  86. if (nread == 0)
  87. break;
  88. ASSERT(nread == sizeof(KeyEvent));
  89. screen.on_receive_keyboard_data(event);
  90. }
  91. }
  92. static WSWindowType from_api(WSAPI_WindowType api_type)
  93. {
  94. switch (api_type) {
  95. case WSAPI_WindowType::Normal:
  96. return WSWindowType::Normal;
  97. case WSAPI_WindowType::Menu:
  98. return WSWindowType::Menu;
  99. case WSAPI_WindowType::WindowSwitcher:
  100. return WSWindowType::WindowSwitcher;
  101. case WSAPI_WindowType::Taskbar:
  102. return WSWindowType::Taskbar;
  103. case WSAPI_WindowType::Tooltip:
  104. return WSWindowType::Tooltip;
  105. default:
  106. ASSERT_NOT_REACHED();
  107. }
  108. }
  109. static Vector<Rect, 32> get_rects(const WSAPI_ClientMessage& message, const ByteBuffer& extra_data)
  110. {
  111. Vector<Rect, 32> rects;
  112. if (message.rect_count > (WSAPI_ClientMessage::max_inline_rect_count + extra_data.size() / sizeof(WSAPI_Rect))) {
  113. return { };
  114. }
  115. for (int i = 0; i < min(WSAPI_ClientMessage::max_inline_rect_count, message.rect_count); ++i)
  116. rects.append(message.rects[i]);
  117. if (!extra_data.is_empty()) {
  118. auto* extra_rects = reinterpret_cast<const WSAPI_Rect*>(extra_data.data());
  119. for (int i = 0; i < (extra_data.size() / sizeof(WSAPI_Rect)); ++i)
  120. rects.append(extra_rects[i]);
  121. }
  122. return rects;
  123. }
  124. bool WSEventLoop::on_receive_from_client(int client_id, const WSAPI_ClientMessage& message, ByteBuffer&& extra_data)
  125. {
  126. WSClientConnection& client = *WSClientConnection::from_client_id(client_id);
  127. switch (message.type) {
  128. case WSAPI_ClientMessage::Type::Greeting:
  129. client.set_client_pid(message.greeting.client_pid);
  130. break;
  131. case WSAPI_ClientMessage::Type::CreateMenubar:
  132. post_event(client, make<WSAPICreateMenubarRequest>(client_id));
  133. break;
  134. case WSAPI_ClientMessage::Type::DestroyMenubar:
  135. post_event(client, make<WSAPIDestroyMenubarRequest>(client_id, message.menu.menubar_id));
  136. break;
  137. case WSAPI_ClientMessage::Type::SetApplicationMenubar:
  138. post_event(client, make<WSAPISetApplicationMenubarRequest>(client_id, message.menu.menubar_id));
  139. break;
  140. case WSAPI_ClientMessage::Type::AddMenuToMenubar:
  141. post_event(client, make<WSAPIAddMenuToMenubarRequest>(client_id, message.menu.menubar_id, message.menu.menu_id));
  142. break;
  143. case WSAPI_ClientMessage::Type::CreateMenu:
  144. ASSERT(message.text_length < (ssize_t)sizeof(message.text));
  145. post_event(client, make<WSAPICreateMenuRequest>(client_id, String(message.text, message.text_length)));
  146. break;
  147. case WSAPI_ClientMessage::Type::PopupMenu:
  148. post_event(client, make<WSAPIPopupMenuRequest>(client_id, message.menu.menu_id, message.menu.position));
  149. break;
  150. case WSAPI_ClientMessage::Type::DismissMenu:
  151. post_event(client, make<WSAPIDismissMenuRequest>(client_id, message.menu.menu_id));
  152. break;
  153. case WSAPI_ClientMessage::Type::SetWindowIcon:
  154. ASSERT(message.text_length < (ssize_t)sizeof(message.text));
  155. post_event(client, make<WSAPISetWindowIconRequest>(client_id, message.window_id, String(message.text, message.text_length)));
  156. break;
  157. case WSAPI_ClientMessage::Type::DestroyMenu:
  158. post_event(client, make<WSAPIDestroyMenuRequest>(client_id, message.menu.menu_id));
  159. break;
  160. case WSAPI_ClientMessage::Type::AddMenuItem:
  161. ASSERT(message.text_length < (ssize_t)sizeof(message.text));
  162. ASSERT(message.menu.shortcut_text_length < (ssize_t)sizeof(message.menu.shortcut_text));
  163. post_event(client, make<WSAPIAddMenuItemRequest>(client_id, message.menu.menu_id, message.menu.identifier, String(message.text, message.text_length), String(message.menu.shortcut_text, message.menu.shortcut_text_length), message.menu.enabled, message.menu.checkable, message.menu.checked));
  164. break;
  165. case WSAPI_ClientMessage::Type::UpdateMenuItem:
  166. ASSERT(message.text_length < (ssize_t)sizeof(message.text));
  167. ASSERT(message.menu.shortcut_text_length < (ssize_t)sizeof(message.menu.shortcut_text));
  168. post_event(client, make<WSAPIUpdateMenuItemRequest>(client_id, message.menu.menu_id, message.menu.identifier, String(message.text, message.text_length), String(message.menu.shortcut_text, message.menu.shortcut_text_length), message.menu.enabled, message.menu.checkable, message.menu.checked));
  169. break;
  170. case WSAPI_ClientMessage::Type::AddMenuSeparator:
  171. post_event(client, make<WSAPIAddMenuSeparatorRequest>(client_id, message.menu.menu_id));
  172. break;
  173. case WSAPI_ClientMessage::Type::CreateWindow:
  174. ASSERT(message.text_length < (ssize_t)sizeof(message.text));
  175. post_event(client, make<WSAPICreateWindowRequest>(client_id, message.window.rect, String(message.text, message.text_length), message.window.has_alpha_channel, message.window.modal, message.window.resizable, message.window.fullscreen, message.window.opacity, message.window.base_size, message.window.size_increment, from_api(message.window.type), Color::from_rgba(message.window.background_color)));
  176. break;
  177. case WSAPI_ClientMessage::Type::DestroyWindow:
  178. post_event(client, make<WSAPIDestroyWindowRequest>(client_id, message.window_id));
  179. break;
  180. case WSAPI_ClientMessage::Type::SetWindowTitle:
  181. ASSERT(message.text_length < (ssize_t)sizeof(message.text));
  182. post_event(client, make<WSAPISetWindowTitleRequest>(client_id, message.window_id, String(message.text, message.text_length)));
  183. break;
  184. case WSAPI_ClientMessage::Type::GetWindowTitle:
  185. ASSERT(message.text_length < (ssize_t)sizeof(message.text));
  186. post_event(client, make<WSAPIGetWindowTitleRequest>(client_id, message.window_id));
  187. break;
  188. case WSAPI_ClientMessage::Type::SetWindowRect:
  189. post_event(client, make<WSAPISetWindowRectRequest>(client_id, message.window_id, message.window.rect));
  190. break;
  191. case WSAPI_ClientMessage::Type::GetWindowRect:
  192. post_event(client, make<WSAPIGetWindowRectRequest>(client_id, message.window_id));
  193. break;
  194. case WSAPI_ClientMessage::Type::SetClipboardContents:
  195. post_event(client, make<WSAPISetClipboardContentsRequest>(client_id, message.clipboard.shared_buffer_id, message.clipboard.contents_size));
  196. break;
  197. case WSAPI_ClientMessage::Type::GetClipboardContents:
  198. post_event(client, make<WSAPIGetClipboardContentsRequest>(client_id));
  199. break;
  200. case WSAPI_ClientMessage::Type::InvalidateRect: {
  201. auto rects = get_rects(message, extra_data);
  202. if (rects.is_empty()) {
  203. client.did_misbehave();
  204. return false;
  205. }
  206. post_event(client, make<WSAPIInvalidateRectRequest>(client_id, message.window_id, rects));
  207. break;
  208. }
  209. case WSAPI_ClientMessage::Type::DidFinishPainting: {
  210. auto rects = get_rects(message, extra_data);
  211. if (rects.is_empty()) {
  212. client.did_misbehave();
  213. return false;
  214. }
  215. post_event(client, make<WSAPIDidFinishPaintingNotification>(client_id, message.window_id, rects));
  216. break;
  217. }
  218. case WSAPI_ClientMessage::Type::GetWindowBackingStore:
  219. post_event(client, make<WSAPIGetWindowBackingStoreRequest>(client_id, message.window_id));
  220. break;
  221. case WSAPI_ClientMessage::Type::SetWindowBackingStore:
  222. post_event(client, make<WSAPISetWindowBackingStoreRequest>(client_id, message.window_id, message.backing.shared_buffer_id, message.backing.size, message.backing.bpp, message.backing.pitch, message.backing.has_alpha_channel, message.backing.flush_immediately));
  223. break;
  224. case WSAPI_ClientMessage::Type::SetGlobalCursorTracking:
  225. post_event(client, make<WSAPISetGlobalCursorTrackingRequest>(client_id, message.window_id, message.value));
  226. break;
  227. case WSAPI_ClientMessage::Type::SetWallpaper:
  228. ASSERT(message.text_length < (ssize_t)sizeof(message.text));
  229. post_event(client, make<WSAPISetWallpaperRequest>(client_id, String(message.text, message.text_length)));
  230. break;
  231. case WSAPI_ClientMessage::Type::GetWallpaper:
  232. post_event(client, make<WSAPIGetWallpaperRequest>(client_id));
  233. break;
  234. case WSAPI_ClientMessage::Type::SetWindowOverrideCursor:
  235. post_event(client, make<WSAPISetWindowOverrideCursorRequest>(client_id, message.window_id, (WSStandardCursor)message.cursor.cursor));
  236. break;
  237. case WSAPI_ClientMessage::SetWindowHasAlphaChannel:
  238. post_event(client, make<WSAPISetWindowHasAlphaChannelRequest>(client_id, message.window_id, message.value));
  239. break;
  240. case WSAPI_ClientMessage::Type::WM_SetActiveWindow:
  241. post_event(client, make<WSWMAPISetActiveWindowRequest>(client_id, message.wm.client_id, message.wm.window_id));
  242. break;
  243. case WSAPI_ClientMessage::Type::WM_SetWindowMinimized:
  244. post_event(client, make<WSWMAPISetWindowMinimizedRequest>(client_id, message.wm.client_id, message.wm.window_id, message.wm.minimized));
  245. break;
  246. case WSAPI_ClientMessage::Type::WM_StartWindowResize:
  247. post_event(client, make<WSWMAPIStartWindowResizeRequest>(client_id, message.wm.client_id, message.wm.window_id));
  248. break;
  249. default:
  250. break;
  251. }
  252. return true;
  253. }
  254. void WSEventLoop::add_file_descriptors_for_select(fd_set& fds, int& max_fd_added)
  255. {
  256. auto add_fd_to_set = [&max_fd_added] (int fd, auto& set) {
  257. FD_SET(fd, &set);
  258. if (fd > max_fd_added)
  259. max_fd_added = fd;
  260. };
  261. add_fd_to_set(m_keyboard_fd, fds);
  262. add_fd_to_set(m_mouse_fd, fds);
  263. add_fd_to_set(m_server_fd, fds);
  264. WSClientConnection::for_each_client([&] (WSClientConnection& client) {
  265. add_fd_to_set(client.fd(), fds);
  266. });
  267. }
  268. void WSEventLoop::process_file_descriptors_after_select(const fd_set& fds)
  269. {
  270. if (FD_ISSET(m_server_fd, &fds))
  271. drain_server();
  272. if (FD_ISSET(m_keyboard_fd, &fds))
  273. drain_keyboard();
  274. if (FD_ISSET(m_mouse_fd, &fds))
  275. drain_mouse();
  276. WSClientConnection::for_each_client([&] (WSClientConnection& client) {
  277. if (FD_ISSET(client.fd(), &fds))
  278. drain_client(client);
  279. });
  280. }
  281. void WSEventLoop::drain_client(WSClientConnection& client)
  282. {
  283. unsigned messages_received = 0;
  284. for (;;) {
  285. WSAPI_ClientMessage message;
  286. // FIXME: Don't go one message at a time, that's so much context switching, oof.
  287. ssize_t nread = read(client.fd(), &message, sizeof(WSAPI_ClientMessage));
  288. if (nread == 0) {
  289. if (!messages_received)
  290. post_event(client, make<WSClientDisconnectedNotification>(client.client_id()));
  291. break;
  292. }
  293. if (nread < 0) {
  294. perror("read");
  295. ASSERT_NOT_REACHED();
  296. }
  297. ByteBuffer extra_data;
  298. if (message.extra_size) {
  299. if (message.extra_size >= 32768) {
  300. dbgprintf("message.extra_size is way too large\n");
  301. return client.did_misbehave();
  302. }
  303. extra_data = ByteBuffer::create_uninitialized(message.extra_size);
  304. fd_set rfds;
  305. FD_ZERO(&rfds);
  306. FD_SET(client.fd(), &rfds);
  307. struct timeval timeout { 1, 0 };
  308. int rc = select(client.fd() + 1, &rfds, nullptr, nullptr, &timeout);
  309. if (rc != 1) {
  310. dbgprintf("extra_data didn't show up in time\n");
  311. return client.did_misbehave();
  312. }
  313. int extra_nread = read(client.fd(), extra_data.data(), extra_data.size());
  314. if (extra_nread != message.extra_size) {
  315. dbgprintf("extra_nread(%d) != extra_size(%d)\n", extra_nread, extra_data.size());
  316. return client.did_misbehave();
  317. }
  318. }
  319. if (!on_receive_from_client(client.client_id(), message, move(extra_data)))
  320. return;
  321. ++messages_received;
  322. }
  323. }