ClientPSMoveAPI Programmer notes - psmoveservice/PSMoveService GitHub Wiki
The client application starts by calling ClientPSMoveAPI::startup()
passing a host address, a port, a callback, and callback userdata. ClientPSMoveAPI
uses the "PImpl Idiom" and its m_implementation_ptr
is instantiated during ::startup()
, passing the arguments along.
Importantly, ClientPSMoveAPIImpl
inherits from IDataFrameListener
, INotificationListener
, and IClientNetworkEventListener
. During instantiation of ClientPSMoveAPIImpl
, it initializes its m_request_manager()
, m_network_manager(see below)
, m_event_callback(callback)
, m_event_callback_userdata(callback_userdata)
, and m_controller_view_map()
.
During initialization of m_network_manager
(an instance of ClientNetworkManager
), ClientPSMoveAPIImpl
passes in itself as the IDataFrameListener
, INotificationListener
, and IClientNetworkEventListener
. m_request_manager
is passed in as the IResponseListener
.
ClientNetworkManager
also uses the PImpl Idiom. During instantiation of ClientNetworkManagerImpl
, we copy pointers to m_data_frame_listener
, m_notification_listener
, m_response_listener
, and m_netEventListener
, among many others. TODO: Need more device types for m_packed_data_frame
.
Then ClientPSMoveAPIImpl::startup()
is called which in turn calls ClientNetworkManager::startup()
which calls ClientNetworkManagerImpl::start()
. This resolves the host&port and calls start_tcp_connect
on the endpoint. start_tcp_connect
creates a tcp connection and passes in ClientNetworkManagerImpl::handle_tcp_connect
as the connection handler which is called when the connect operation completes. Assuming everything goes smoothly, we get the corresponding m_udp_server_endpoint
then we call start_tcp_read_response_header()
.
start_tcp_read_response_header()
is the beginning of a loop, including m_response_read_buffer
, handle_tcp_read_response_header
, start_tcp_read_response_body()
, handle_tcp_read_response_body
, and back to start_tcp_read_response_header()
.
During handle_tcp_read_response_body
, we call handle_tcp_response_received()
. This is a branch point. If the response has a request id then we call m_response_listener->handle_response(response)
. Otherwise, this is a notification, and we call handle_tcp_connection_info_notification
if Response_ResponseType_CONNECTION_INFO
or m_notification_listener->handle_notification(response)
.
Remember that the m_response_listener
is in fact ClientRequestManager
(PImpl). ClientRequestManagerImpl::handle_response
first checks that there's a corresponding entry in m_pending_requests
. If it finds the request, then it gets the RequestContext context
then calls the context.callback
. There would only be an entry in m_pending_requests
if it was put there by the App (e.g., ClientPSMoveAPI::start_controller_data_stream
during AppStage_MagnetometerCalibration::enter()
). Similarly, the callback would have been specified by the app.
Remember that m_notification_listener
is ClientPSMoveAPIImpl
. ClientPSMoveAPIImpl::handle_notification
calls m_event_callback
. m_event_callback
is defined by the App (e.g., App::onClientPSMoveEvent
. The first argument to m_event_callback
is the ClientPSMoveAPI::eClientPSMoveAPIEvent
eventType. Currently the only specific eventTypes handled is Response_ResponseType_CONTROLLER_LIST_UPDATED
. TODO: Different event types for different devices. If the app-defined callback then handles the notification, maybe checking the event type (e.g., this might trigger a request_controller_list()
).
ClientNetworkManagerImpl::handle_tcp_connection_info_notification
starts a chain -> ClientNetworkManagerImpl::send_udp_connection_id()
-> ClientNetworkManagerImpl::handle_udp_write_connection_id
-> ClientNetworkManagerImpl::handle_udp_read_connection_result
. If no errors, this calls 3 methods: start_udp_read_data_frame()
, start_tcp_write_request()
, and m_netEventListener->handle_server_connection_opened()
.
ClientNetworkManagerImpl::start_udp_read_data_frame()
does an async receive from UDP with handler ClientNetworkManagerImpl::handle_udp_read_data_frame
. This handler calls handle_udp_data_frame_received()
then goes back to start_udp_read_data_frame()
to read the next frame. handle_udp_data_frame_received()
gets the data frame then passes it off to (ClientPSMoveAPIPImpl
) m_data_frame_listener->handle_data_frame
. Currently this assumes the return of the data .get_msg()
is ControllerDataFramePtr
. TODO: Check for data frame type, get the correct type, then call overloaded handle_data_frame
.
ClientPSMoveAPIPImpl::handle_data_frame
is device-type specific. TODO: create more handle_data_frame
functions for different data_frame types. This gets the ClientControllerView
, and ApplyControllerDataFrame
. This copies information from the data_frame to the instance of the Client<Device>View
.
Remember that m_netEventListener
is ClientPSMoveAPIPImpl
. ClientPSMoveAPIPImpl::handle_server_connection_opened()
calls the application event callback with event type ClientPSMoveAPI::connectedToService
.
On each loop of the client application, it calls ClientPSMoveAPI::update()
and maybe controller_view->GetDataFrameFPS()
.
ClientPSMoveAPI::update()
calls m_network_manager.update()
, and that's it.
m_network_manager
is an instance of ClientNetworkManager
. Its .update()
calls m_io_service.poll()
, where m_io_service
is an instance of boost::asio::io_service
and poll()
runs all the handlers that are bound to the sockets (I think).