| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  | #include <LibAudio/ASAPI.h>
 | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  | #include <LibCore/CEvent.h>
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  | #include <LibCore/CEventLoop.h>
 | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  | #include <LibCore/CLocalSocket.h>
 | 
					
						
							|  |  |  | #include <LibCore/CNotifier.h>
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  | #include <LibCore/CSyscallUtils.h>
 | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <sys/select.h>
 | 
					
						
							|  |  |  | #include <sys/socket.h>
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  | #include <sys/types.h>
 | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  | #include <sys/uio.h>
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  | //#define CIPC_DEBUG
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  | namespace IPC { | 
					
						
							| 
									
										
										
										
											2019-07-17 19:46:06 +02:00
										 |  |  | namespace Client { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |     class Event : public CEvent { | 
					
						
							|  |  |  |     public: | 
					
						
							|  |  |  |         enum Type { | 
					
						
							|  |  |  |             Invalid = 2000, | 
					
						
							|  |  |  |             PostProcess, | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  |         }; | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |         Event() {} | 
					
						
							|  |  |  |         explicit Event(Type type) | 
					
						
							|  |  |  |             : CEvent(type) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |     class PostProcessEvent : public Event { | 
					
						
							|  |  |  |     public: | 
					
						
							|  |  |  |         explicit PostProcessEvent(int client_id) | 
					
						
							|  |  |  |             : Event(PostProcess) | 
					
						
							|  |  |  |             , m_client_id(client_id) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |         int client_id() const { return m_client_id; } | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |     private: | 
					
						
							|  |  |  |         int m_client_id { 0 }; | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |     template<typename ServerMessage, typename ClientMessage> | 
					
						
							|  |  |  |     class Connection : public CObject { | 
					
						
							|  |  |  |         C_OBJECT(Connection) | 
					
						
							|  |  |  |     public: | 
					
						
							|  |  |  |         Connection(const StringView& address) | 
					
						
							|  |  |  |             : m_notifier(CNotifier(m_connection.fd(), CNotifier::Read)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // We want to rate-limit our clients
 | 
					
						
							|  |  |  |             m_connection.set_blocking(true); | 
					
						
							|  |  |  |             m_notifier.on_ready_to_read = [this] { | 
					
						
							|  |  |  |                 drain_messages_from_server(); | 
					
						
							| 
									
										
										
										
											2019-07-17 19:46:06 +02:00
										 |  |  |                 CEventLoop::current().post_event(*this, make<PostProcessEvent>(m_connection.fd())); | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |             }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             int retries = 1000; | 
					
						
							|  |  |  |             while (retries) { | 
					
						
							|  |  |  |                 if (m_connection.connect(CSocketAddress::local(address))) { | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 dbgprintf("Client::Connection: connect failed: %d, %s\n", errno, strerror(errno)); | 
					
						
							|  |  |  |                 sleep(1); | 
					
						
							|  |  |  |                 --retries; | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |             ASSERT(m_connection.is_connected()); | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         virtual void handshake() = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         virtual void event(CEvent& event) override | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (event.type() == Event::PostProcess) { | 
					
						
							|  |  |  |                 postprocess_bundles(m_unprocessed_bundles); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 CObject::event(event); | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         void set_server_pid(pid_t pid) { m_server_pid = pid; } | 
					
						
							|  |  |  |         pid_t server_pid() const { return m_server_pid; } | 
					
						
							|  |  |  |         void set_my_client_id(int id) { m_my_client_id = id; } | 
					
						
							|  |  |  |         int my_client_id() const { return m_my_client_id; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         template<typename MessageType> | 
					
						
							|  |  |  |         bool wait_for_specific_event(MessageType type, ServerMessage& event) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // Double check we don't already have the event waiting for us.
 | 
					
						
							|  |  |  |             // Otherwise we might end up blocked for a while for no reason.
 | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  |             for (ssize_t i = 0; i < m_unprocessed_bundles.size(); ++i) { | 
					
						
							|  |  |  |                 if (m_unprocessed_bundles[i].message.type == type) { | 
					
						
							|  |  |  |                     event = move(m_unprocessed_bundles[i].message); | 
					
						
							|  |  |  |                     m_unprocessed_bundles.remove(i); | 
					
						
							| 
									
										
										
										
											2019-07-17 19:46:06 +02:00
										 |  |  |                     CEventLoop::current().post_event(*this, make<PostProcessEvent>(m_connection.fd())); | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  |                     return true; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |             for (;;) { | 
					
						
							|  |  |  |                 fd_set rfds; | 
					
						
							|  |  |  |                 FD_ZERO(&rfds); | 
					
						
							|  |  |  |                 FD_SET(m_connection.fd(), &rfds); | 
					
						
							|  |  |  |                 int rc = CSyscallUtils::safe_syscall(select, m_connection.fd() + 1, &rfds, nullptr, nullptr, nullptr); | 
					
						
							|  |  |  |                 if (rc < 0) { | 
					
						
							|  |  |  |                     perror("select"); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 ASSERT(rc > 0); | 
					
						
							|  |  |  |                 ASSERT(FD_ISSET(m_connection.fd(), &rfds)); | 
					
						
							|  |  |  |                 bool success = drain_messages_from_server(); | 
					
						
							|  |  |  |                 if (!success) | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |                 for (ssize_t i = 0; i < m_unprocessed_bundles.size(); ++i) { | 
					
						
							|  |  |  |                     if (m_unprocessed_bundles[i].message.type == type) { | 
					
						
							|  |  |  |                         event = move(m_unprocessed_bundles[i].message); | 
					
						
							|  |  |  |                         m_unprocessed_bundles.remove(i); | 
					
						
							|  |  |  |                         CEventLoop::current().post_event(*this, make<PostProcessEvent>(m_connection.fd())); | 
					
						
							|  |  |  |                         return true; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |         bool post_message_to_server(const ClientMessage& message, const ByteBuffer&& extra_data = {}) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  | #if defined(CIPC_DEBUG)
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |             dbg() << "C: -> S " << int(message.type) << " extra " << extra_data.size(); | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |             if (!extra_data.is_empty()) | 
					
						
							|  |  |  |                 const_cast<ClientMessage&>(message).extra_size = extra_data.size(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             struct iovec iov[2]; | 
					
						
							|  |  |  |             int iov_count = 1; | 
					
						
							|  |  |  |             iov[0].iov_base = const_cast<ClientMessage*>(&message); | 
					
						
							|  |  |  |             iov[0].iov_len = sizeof(message); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!extra_data.is_empty()) { | 
					
						
							|  |  |  |                 iov[1].iov_base = const_cast<u8*>(extra_data.data()); | 
					
						
							|  |  |  |                 iov[1].iov_len = extra_data.size(); | 
					
						
							|  |  |  |                 ++iov_count; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             int nwritten = writev(m_connection.fd(), iov, iov_count); | 
					
						
							|  |  |  |             if (nwritten < 0) { | 
					
						
							|  |  |  |                 perror("writev"); | 
					
						
							|  |  |  |                 ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             ASSERT((size_t)nwritten == sizeof(message) + extra_data.size()); | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |         template<typename MessageType> | 
					
						
							|  |  |  |         ServerMessage sync_request(const ClientMessage& request, MessageType response_type) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             bool success = post_message_to_server(request); | 
					
						
							|  |  |  |             ASSERT(success); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             ServerMessage response; | 
					
						
							|  |  |  |             success = wait_for_specific_event(response_type, response); | 
					
						
							|  |  |  |             ASSERT(success); | 
					
						
							|  |  |  |             return response; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |     protected: | 
					
						
							|  |  |  |         struct IncomingMessageBundle { | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  |             ServerMessage message; | 
					
						
							|  |  |  |             ByteBuffer extra_data; | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         virtual void postprocess_bundles(Vector<IncomingMessageBundle>& new_bundles) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             dbg() << "Client::Connection: " | 
					
						
							|  |  |  |                   << " warning: discarding " << new_bundles.size() << " unprocessed bundles; this may not be what you want"; | 
					
						
							|  |  |  |             new_bundles.clear(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private: | 
					
						
							|  |  |  |         bool drain_messages_from_server() | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             for (;;) { | 
					
						
							|  |  |  |                 ServerMessage message; | 
					
						
							|  |  |  |                 ssize_t nread = recv(m_connection.fd(), &message, sizeof(ServerMessage), MSG_DONTWAIT); | 
					
						
							|  |  |  |                 if (nread < 0) { | 
					
						
							|  |  |  |                     if (errno == EAGAIN) { | 
					
						
							|  |  |  |                         return true; | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  |                     perror("read"); | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |                     exit(1); | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (nread == 0) { | 
					
						
							|  |  |  |                     dbgprintf("EOF on IPC fd\n"); | 
					
						
							|  |  |  |                     exit(1); | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 ASSERT(nread == sizeof(message)); | 
					
						
							|  |  |  |                 ByteBuffer extra_data; | 
					
						
							|  |  |  |                 if (message.extra_size) { | 
					
						
							|  |  |  |                     extra_data = ByteBuffer::create_uninitialized(message.extra_size); | 
					
						
							|  |  |  |                     int extra_nread = read(m_connection.fd(), extra_data.data(), extra_data.size()); | 
					
						
							|  |  |  |                     if (extra_nread < 0) { | 
					
						
							|  |  |  |                         perror("read"); | 
					
						
							|  |  |  |                         ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     ASSERT((size_t)extra_nread == message.extra_size); | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  | #if defined(CIPC_DEBUG)
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |                 dbg() << "C: <- S " << int(message.type) << " extra " << extra_data.size(); | 
					
						
							| 
									
										
										
										
											2019-07-17 18:28:30 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |                 m_unprocessed_bundles.append({ move(message), move(extra_data) }); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-07-17 14:52:12 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-27 10:50:26 +02:00
										 |  |  |         CLocalSocket m_connection; | 
					
						
							|  |  |  |         CNotifier m_notifier; | 
					
						
							|  |  |  |         Vector<IncomingMessageBundle> m_unprocessed_bundles; | 
					
						
							|  |  |  |         int m_server_pid; | 
					
						
							|  |  |  |         int m_my_client_id; | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2019-07-17 19:46:06 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | } // Client
 | 
					
						
							|  |  |  | } // IPC
 |