CapoDetector.cpp 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. /*
  2. * Copyright 2022 Google LLC. All Rights Reserved.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. #include "CapoDetector.h"
  17. #include <google/protobuf/message.h>
  18. #include <google/protobuf/io/coded_stream.h>
  19. #include <google/protobuf/io/zero_copy_stream_impl.h>
  20. #include <log/log.h>
  21. #ifdef LOG_TAG
  22. #undef LOG_TAG
  23. #define LOG_TAG "CapoDetector"
  24. #endif
  25. namespace android {
  26. namespace chre {
  27. namespace { // anonymous namespace for file-local definitions
  28. static capo::ConfigureDetector_ConfigData config_data = capo::ConfigureDetector_ConfigData();
  29. static capo::ConfigureDetector msg = capo::ConfigureDetector();
  30. /**
  31. * Called when onConnected() to send NanoappList request.
  32. */
  33. void requestNanoappList(SocketClient &client) {
  34. flatbuffers::FlatBufferBuilder builder;
  35. HostProtocolHost::encodeNanoappListRequest(builder);
  36. if (!client.sendMessage(builder.GetBufferPointer(), builder.GetSize())) {
  37. ALOGE("Failed to send NanoappList request");
  38. }
  39. }
  40. } // namespace
  41. /**
  42. * Called when initializing connection with CHRE socket.
  43. */
  44. sp<CapoDetector> CapoDetector::start() {
  45. sp<CapoDetector> listener = new CapoDetector();
  46. if (!listener->connectInBackground(kChreSocketName, listener)) {
  47. ALOGE("Couldn't connect to CHRE socket");
  48. return nullptr;
  49. }
  50. ALOGI("%s connect to CHRE socket.", __func__);
  51. return listener;
  52. }
  53. /**
  54. * Called when the socket is successfully (re-)connected.
  55. * Reset the position and try to send NanoappList request.
  56. */
  57. void CapoDetector::onConnected() {
  58. flatbuffers::FlatBufferBuilder builder;
  59. // Reset the last position type.
  60. last_position_type_ = capo::PositionType::UNKNOWN;
  61. requestNanoappList(*this);
  62. }
  63. /**
  64. * Called when we have failed to (re-)connect the socket after many attempts
  65. * and are giving up.
  66. */
  67. void CapoDetector::onConnectionAborted() {
  68. ALOGE("%s, Capo Aborting Connection!", __func__);
  69. }
  70. /**
  71. * Invoked when the socket is disconnected, and this connection loss was not
  72. * the result of an explicit call to disconnect().
  73. * Reset the position while disconnecting.
  74. */
  75. void CapoDetector::onDisconnected() {
  76. last_position_type_ = capo::PositionType::UNKNOWN;
  77. }
  78. /**
  79. * Decode unix socket msgs to CHRE messages, and call the appropriate
  80. * callback depending on the CHRE message.
  81. */
  82. void CapoDetector::onMessageReceived(const void *data, size_t length) {
  83. if (!HostProtocolHost::decodeMessageFromChre(data, length, *this)) {
  84. ALOGE("Failed to decode message");
  85. }
  86. }
  87. /**
  88. * Listen for messages from capo nanoapp and handle the message.
  89. */
  90. void CapoDetector::handleNanoappMessage(const fbs::NanoappMessageT &message) {
  91. ALOGI("%s, Id %" PRIu64 ", type %d, size %d", __func__, message.app_id, message.message_type,
  92. static_cast<int>(message.message.size()));
  93. // Exclude the message with unmatched nanoapp id.
  94. if (message.app_id != kCapoNanoappId)
  95. return;
  96. // Handle the message with message_type.
  97. switch (message.message_type) {
  98. case capo::MessageType::ACK_NOTIFICATION: {
  99. capo::AckNotification gd;
  100. gd.set_notification_type(static_cast<capo::NotificationType>(message.message[1]));
  101. ALOGD("%s, get notification event from capo nanoapp, type %d", __func__,
  102. gd.notification_type());
  103. break;
  104. }
  105. case capo::MessageType::POSITION_DETECTED: {
  106. capo::PositionDetected gd;
  107. gd.set_position_type(static_cast<capo::PositionType>(message.message[1]));
  108. ALOGD("%s, get position event from capo nanoapp, type %d", __func__,
  109. gd.position_type());
  110. // Callback to function while getting carried position event.
  111. if (callback_func_ != nullptr) {
  112. last_position_type_ = gd.position_type();
  113. ALOGD("%s, sent position type %d to callback function", __func__,
  114. last_position_type_);
  115. callback_func_(last_position_type_);
  116. }
  117. break;
  118. }
  119. default:
  120. ALOGE("%s, get invalid message, type: %" PRIu32 ", from capo nanoapp.", __func__,
  121. message.message_type);
  122. break;
  123. }
  124. }
  125. /**
  126. * Handle the response of a NanoappList request.
  127. * Ensure that capo nanoapp is running.
  128. */
  129. void CapoDetector::handleNanoappListResponse(const fbs::NanoappListResponseT &response) {
  130. for (const std::unique_ptr<fbs::NanoappListEntryT> &nanoapp : response.nanoapps) {
  131. if (nanoapp->app_id == kCapoNanoappId) {
  132. if (nanoapp->enabled)
  133. enable();
  134. else
  135. ALOGE("Capo nanoapp not enabled");
  136. return;
  137. }
  138. }
  139. ALOGE("Capo nanoapp not found");
  140. }
  141. /**
  142. * Send enabling message to the nanoapp.
  143. */
  144. void CapoDetector::enable() {
  145. // Create CHRE message with serialized message
  146. flatbuffers::FlatBufferBuilder builder, config_builder, force_builder;
  147. config_data.set_still_time_threshold_nanosecond(mCapoDetectorMDParameters.still_time_threshold_ns);
  148. config_data.set_window_width_nanosecond(mCapoDetectorMDParameters.window_width_ns);
  149. config_data.set_motion_confidence_threshold(mCapoDetectorMDParameters.motion_confidence_threshold);
  150. config_data.set_still_confidence_threshold(mCapoDetectorMDParameters.still_confidence_threshold);
  151. config_data.set_var_threshold(mCapoDetectorMDParameters.var_threshold);
  152. config_data.set_var_threshold_delta(mCapoDetectorMDParameters.var_threshold_delta);
  153. msg.set_allocated_config_data(&config_data);
  154. auto pb_size = msg.ByteSizeLong();
  155. auto pb_data = std::make_unique<uint8_t[]>(pb_size);
  156. if (!msg.SerializeToArray(pb_data.get(), pb_size)) {
  157. ALOGE("Failed to serialize message.");
  158. }
  159. ALOGI("Configuring CapoDetector");
  160. // Configure the detector from host-side
  161. android::chre::HostProtocolHost::encodeNanoappMessage(
  162. config_builder, getNanoppAppId(), capo::MessageType::CONFIGURE_DETECTOR, getHostEndPoint(),
  163. pb_data.get(), pb_size);
  164. ALOGI("Sending capo config message to Nanoapp, %" PRIu32 " bytes", config_builder.GetSize());
  165. if (!sendMessage(config_builder.GetBufferPointer(), config_builder.GetSize())) {
  166. ALOGE("Failed to send config event for capo nanoapp");
  167. }
  168. ALOGI("Enabling CapoDetector");
  169. android::chre::HostProtocolHost::encodeNanoappMessage(
  170. builder, getNanoppAppId(), capo::MessageType::ENABLE_DETECTOR, getHostEndPoint(),
  171. /*messageData*/ nullptr, /*messageDataLenbuffer*/ 0);
  172. ALOGI("Sending enable message to Nanoapp, %" PRIu32 " bytes", builder.GetSize());
  173. if (!sendMessage(builder.GetBufferPointer(), builder.GetSize())) {
  174. ALOGE("Failed to send enable event for capo nanoapp");
  175. }
  176. ALOGI("Forcing CapoDetector to update state");
  177. // Force an updated state upon connection
  178. android::chre::HostProtocolHost::encodeNanoappMessage(
  179. force_builder, getNanoppAppId(), capo::MessageType::FORCE_UPDATE, getHostEndPoint(),
  180. /*messageData*/ nullptr, /*messageDataLenbuffer*/ 0);
  181. ALOGI("Sending force-update message to Nanoapp, %" PRIu32 " bytes", force_builder.GetSize());
  182. if (!sendMessage(force_builder.GetBufferPointer(), force_builder.GetSize())) {
  183. ALOGE("Failed to send force-update event for capo nanoapp");
  184. }
  185. }
  186. } // namespace chre
  187. } // namespace android