WSEventLoop.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371
  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_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. dbgprintf("Unknown WSAPI_WindowType: %d\n", api_type);
  107. ASSERT_NOT_REACHED();
  108. }
  109. }
  110. static Vector<Rect, 32> get_rects(const WSAPI_ClientMessage& message, const ByteBuffer& extra_data)
  111. {
  112. Vector<Rect, 32> rects;
  113. if (message.rect_count > (WSAPI_ClientMessage::max_inline_rect_count + extra_data.size() / sizeof(WSAPI_Rect))) {
  114. return { };
  115. }
  116. for (int i = 0; i < min(WSAPI_ClientMessage::max_inline_rect_count, message.rect_count); ++i)
  117. rects.append(message.rects[i]);
  118. if (!extra_data.is_empty()) {
  119. auto* extra_rects = reinterpret_cast<const WSAPI_Rect*>(extra_data.data());
  120. for (int i = 0; i < (extra_data.size() / sizeof(WSAPI_Rect)); ++i)
  121. rects.append(extra_rects[i]);
  122. }
  123. return rects;
  124. }
  125. bool WSEventLoop::on_receive_from_client(int client_id, const WSAPI_ClientMessage& message, ByteBuffer&& extra_data)
  126. {
  127. WSClientConnection& client = *WSClientConnection::from_client_id(client_id);
  128. switch (message.type) {
  129. case WSAPI_ClientMessage::Type::Greeting:
  130. client.set_client_pid(message.greeting.client_pid);
  131. break;
  132. case WSAPI_ClientMessage::Type::CreateMenubar:
  133. post_event(client, make<WSAPICreateMenubarRequest>(client_id));
  134. break;
  135. case WSAPI_ClientMessage::Type::DestroyMenubar:
  136. post_event(client, make<WSAPIDestroyMenubarRequest>(client_id, message.menu.menubar_id));
  137. break;
  138. case WSAPI_ClientMessage::Type::SetApplicationMenubar:
  139. post_event(client, make<WSAPISetApplicationMenubarRequest>(client_id, message.menu.menubar_id));
  140. break;
  141. case WSAPI_ClientMessage::Type::AddMenuToMenubar:
  142. post_event(client, make<WSAPIAddMenuToMenubarRequest>(client_id, message.menu.menubar_id, message.menu.menu_id));
  143. break;
  144. case WSAPI_ClientMessage::Type::CreateMenu:
  145. if (message.text_length > (int)sizeof(message.text)) {
  146. client.did_misbehave();
  147. return false;
  148. }
  149. post_event(client, make<WSAPICreateMenuRequest>(client_id, String(message.text, message.text_length)));
  150. break;
  151. case WSAPI_ClientMessage::Type::PopupMenu:
  152. post_event(client, make<WSAPIPopupMenuRequest>(client_id, message.menu.menu_id, message.menu.position));
  153. break;
  154. case WSAPI_ClientMessage::Type::DismissMenu:
  155. post_event(client, make<WSAPIDismissMenuRequest>(client_id, message.menu.menu_id));
  156. break;
  157. case WSAPI_ClientMessage::Type::SetWindowIcon:
  158. if (message.text_length > (int)sizeof(message.text)) {
  159. client.did_misbehave();
  160. return false;
  161. }
  162. post_event(client, make<WSAPISetWindowIconRequest>(client_id, message.window_id, String(message.text, message.text_length)));
  163. break;
  164. case WSAPI_ClientMessage::Type::DestroyMenu:
  165. post_event(client, make<WSAPIDestroyMenuRequest>(client_id, message.menu.menu_id));
  166. break;
  167. case WSAPI_ClientMessage::Type::AddMenuItem:
  168. if (message.text_length > (int)sizeof(message.text)) {
  169. client.did_misbehave();
  170. return false;
  171. }
  172. if (message.menu.shortcut_text_length > (int)sizeof(message.menu.shortcut_text)) {
  173. client.did_misbehave();
  174. return false;
  175. }
  176. 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));
  177. break;
  178. case WSAPI_ClientMessage::Type::UpdateMenuItem:
  179. if (message.text_length > (int)sizeof(message.text)) {
  180. client.did_misbehave();
  181. return false;
  182. }
  183. if (message.menu.shortcut_text_length > (int)sizeof(message.menu.shortcut_text)) {
  184. client.did_misbehave();
  185. return false;
  186. }
  187. 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));
  188. break;
  189. case WSAPI_ClientMessage::Type::AddMenuSeparator:
  190. post_event(client, make<WSAPIAddMenuSeparatorRequest>(client_id, message.menu.menu_id));
  191. break;
  192. case WSAPI_ClientMessage::Type::CreateWindow:
  193. if (message.text_length > (int)sizeof(message.text)) {
  194. client.did_misbehave();
  195. return false;
  196. }
  197. post_event(client,
  198. make<WSAPICreateWindowRequest>(client_id,
  199. message.window.rect,
  200. String(message.text, message.text_length),
  201. message.window.has_alpha_channel,
  202. message.window.modal,
  203. message.window.resizable,
  204. message.window.fullscreen,
  205. message.window.show_titlebar,
  206. message.window.opacity,
  207. message.window.base_size,
  208. message.window.size_increment,
  209. from_api(message.window.type),
  210. Color::from_rgba(message.window.background_color)));
  211. break;
  212. case WSAPI_ClientMessage::Type::DestroyWindow:
  213. post_event(client, make<WSAPIDestroyWindowRequest>(client_id, message.window_id));
  214. break;
  215. case WSAPI_ClientMessage::Type::SetWindowTitle:
  216. if (message.text_length > (int)sizeof(message.text)) {
  217. client.did_misbehave();
  218. return false;
  219. }
  220. post_event(client, make<WSAPISetWindowTitleRequest>(client_id, message.window_id, String(message.text, message.text_length)));
  221. break;
  222. case WSAPI_ClientMessage::Type::GetWindowTitle:
  223. post_event(client, make<WSAPIGetWindowTitleRequest>(client_id, message.window_id));
  224. break;
  225. case WSAPI_ClientMessage::Type::SetWindowRect:
  226. post_event(client, make<WSAPISetWindowRectRequest>(client_id, message.window_id, message.window.rect));
  227. break;
  228. case WSAPI_ClientMessage::Type::GetWindowRect:
  229. post_event(client, make<WSAPIGetWindowRectRequest>(client_id, message.window_id));
  230. break;
  231. case WSAPI_ClientMessage::Type::SetClipboardContents:
  232. post_event(client, make<WSAPISetClipboardContentsRequest>(client_id, message.clipboard.shared_buffer_id, message.clipboard.contents_size));
  233. break;
  234. case WSAPI_ClientMessage::Type::GetClipboardContents:
  235. post_event(client, make<WSAPIGetClipboardContentsRequest>(client_id));
  236. break;
  237. case WSAPI_ClientMessage::Type::InvalidateRect: {
  238. auto rects = get_rects(message, extra_data);
  239. if (rects.is_empty()) {
  240. client.did_misbehave();
  241. return false;
  242. }
  243. post_event(client, make<WSAPIInvalidateRectRequest>(client_id, message.window_id, rects));
  244. break;
  245. }
  246. case WSAPI_ClientMessage::Type::DidFinishPainting: {
  247. auto rects = get_rects(message, extra_data);
  248. if (rects.is_empty()) {
  249. client.did_misbehave();
  250. return false;
  251. }
  252. post_event(client, make<WSAPIDidFinishPaintingNotification>(client_id, message.window_id, rects));
  253. break;
  254. }
  255. case WSAPI_ClientMessage::Type::GetWindowBackingStore:
  256. post_event(client, make<WSAPIGetWindowBackingStoreRequest>(client_id, message.window_id));
  257. break;
  258. case WSAPI_ClientMessage::Type::SetWindowBackingStore:
  259. 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));
  260. break;
  261. case WSAPI_ClientMessage::Type::SetGlobalCursorTracking:
  262. post_event(client, make<WSAPISetGlobalCursorTrackingRequest>(client_id, message.window_id, message.value));
  263. break;
  264. case WSAPI_ClientMessage::Type::SetWallpaper:
  265. if (message.text_length > (int)sizeof(message.text)) {
  266. client.did_misbehave();
  267. return false;
  268. }
  269. post_event(client, make<WSAPISetWallpaperRequest>(client_id, String(message.text, message.text_length)));
  270. break;
  271. case WSAPI_ClientMessage::Type::GetWallpaper:
  272. post_event(client, make<WSAPIGetWallpaperRequest>(client_id));
  273. break;
  274. case WSAPI_ClientMessage::Type::SetWindowOverrideCursor:
  275. post_event(client, make<WSAPISetWindowOverrideCursorRequest>(client_id, message.window_id, (WSStandardCursor)message.cursor.cursor));
  276. break;
  277. case WSAPI_ClientMessage::SetWindowHasAlphaChannel:
  278. post_event(client, make<WSAPISetWindowHasAlphaChannelRequest>(client_id, message.window_id, message.value));
  279. break;
  280. case WSAPI_ClientMessage::Type::WM_SetActiveWindow:
  281. post_event(client, make<WSWMAPISetActiveWindowRequest>(client_id, message.wm.client_id, message.wm.window_id));
  282. break;
  283. case WSAPI_ClientMessage::Type::WM_SetWindowMinimized:
  284. post_event(client, make<WSWMAPISetWindowMinimizedRequest>(client_id, message.wm.client_id, message.wm.window_id, message.wm.minimized));
  285. break;
  286. case WSAPI_ClientMessage::Type::WM_StartWindowResize:
  287. post_event(client, make<WSWMAPIStartWindowResizeRequest>(client_id, message.wm.client_id, message.wm.window_id));
  288. break;
  289. default:
  290. break;
  291. }
  292. return true;
  293. }
  294. void WSEventLoop::add_file_descriptors_for_select(fd_set& fds, int& max_fd_added)
  295. {
  296. auto add_fd_to_set = [&max_fd_added] (int fd, auto& set) {
  297. FD_SET(fd, &set);
  298. if (fd > max_fd_added)
  299. max_fd_added = fd;
  300. };
  301. add_fd_to_set(m_keyboard_fd, fds);
  302. add_fd_to_set(m_mouse_fd, fds);
  303. add_fd_to_set(m_server_fd, fds);
  304. WSClientConnection::for_each_client([&] (WSClientConnection& client) {
  305. add_fd_to_set(client.fd(), fds);
  306. });
  307. }
  308. void WSEventLoop::process_file_descriptors_after_select(const fd_set& fds)
  309. {
  310. if (FD_ISSET(m_server_fd, &fds))
  311. drain_server();
  312. if (FD_ISSET(m_keyboard_fd, &fds))
  313. drain_keyboard();
  314. if (FD_ISSET(m_mouse_fd, &fds))
  315. drain_mouse();
  316. WSClientConnection::for_each_client([&] (WSClientConnection& client) {
  317. if (FD_ISSET(client.fd(), &fds))
  318. drain_client(client);
  319. });
  320. }
  321. void WSEventLoop::drain_client(WSClientConnection& client)
  322. {
  323. unsigned messages_received = 0;
  324. for (;;) {
  325. WSAPI_ClientMessage message;
  326. // FIXME: Don't go one message at a time, that's so much context switching, oof.
  327. ssize_t nread = recv(client.fd(), &message, sizeof(WSAPI_ClientMessage), MSG_DONTWAIT);
  328. if (nread == 0 || (nread == -1 && errno == EAGAIN)) {
  329. if (!messages_received)
  330. post_event(client, make<WSClientDisconnectedNotification>(client.client_id()));
  331. break;
  332. }
  333. if (nread < 0) {
  334. perror("recv");
  335. ASSERT_NOT_REACHED();
  336. }
  337. ByteBuffer extra_data;
  338. if (message.extra_size) {
  339. if (message.extra_size >= 32768) {
  340. dbgprintf("message.extra_size is way too large\n");
  341. return client.did_misbehave();
  342. }
  343. extra_data = ByteBuffer::create_uninitialized(message.extra_size);
  344. // FIXME: We should allow this to time out. Maybe use a socket timeout?
  345. int extra_nread = read(client.fd(), extra_data.data(), extra_data.size());
  346. if (extra_nread != message.extra_size) {
  347. dbgprintf("extra_nread(%d) != extra_size(%d)\n", extra_nread, extra_data.size());
  348. return client.did_misbehave();
  349. }
  350. }
  351. if (!on_receive_from_client(client.client_id(), message, move(extra_data)))
  352. return;
  353. ++messages_received;
  354. }
  355. }