• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2017 The Android Open Source Project
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 "chre_interface.h"
17 
18 #include <android-base/logging.h>
19 
20 #include <chrono>
21 
22 #include "chre_host/host_protocol_host.h"
23 
24 using android::chre::HostProtocolHost;
25 using flatbuffers::FlatBufferBuilder;
26 
27 namespace chre_constants = android::hardware::wifi::offload::V1_0::implementation::chre_constants;
28 
29 namespace fbs = ::chre::fbs;
30 
31 namespace android {
32 namespace hardware {
33 namespace wifi {
34 namespace offload {
35 namespace V1_0 {
36 namespace implementation {
37 
SocketCallbacks(ChreInterface * parent)38 SocketCallbacks::SocketCallbacks(ChreInterface* parent) : mParent(parent) {
39 }
40 
onMessageReceived(const void * data,size_t length)41 void SocketCallbacks::onMessageReceived(const void* data, size_t length) {
42     LOG(VERBOSE) << "Message received from CHRE socket";
43     if (!HostProtocolHost::decodeMessageFromChre(data, length, *this)) {
44         LOG(WARNING) << "Failed to decode message";
45     }
46 }
47 
onConnected()48 void SocketCallbacks::onConnected() {
49     LOG(INFO) << "Connected to CHRE socket";
50     mParent->reportConnectionEvent(ChreInterfaceCallbacks::CONNECTED);
51 }
52 
onConnectionAborted()53 void SocketCallbacks::onConnectionAborted() {
54     LOG(WARNING) << "Connection to CHRE socket Aborted";
55     mParent->reportConnectionEvent(ChreInterfaceCallbacks::CONNECTION_ABORT);
56 }
57 
onDisconnected()58 void SocketCallbacks::onDisconnected() {
59     LOG(WARNING) << "Disconnected from CHRE socket";
60     mParent->reportConnectionEvent(ChreInterfaceCallbacks::DISCONNECTED);
61 }
62 
handleNanoappMessage(uint64_t appId,uint32_t messageType,uint16_t hostEndpoint,const void * messageData,size_t messageDataLen)63 void SocketCallbacks::handleNanoappMessage(uint64_t appId, uint32_t messageType,
64                                            uint16_t hostEndpoint, const void* messageData,
65                                            size_t messageDataLen) {
66     LOG(VERBOSE) << "handleNanoappMessage from appId: " << appId;
67     LOG(VERBOSE) << "HostEndPoint: " << hostEndpoint;
68     if (appId == chre_constants::kWifiOffloadNanoAppId) {
69         mParent->handleMessage(messageType, messageData, messageDataLen);
70     }
71 }
72 
handleHubInfoResponse(const char * name,const char * vendor,const char * toolchain,uint32_t legacyPlatformVersion,uint32_t legacyToolchainVersion,float peakMips,float stoppedPower,float sleepPower,float peakPower,uint32_t maxMessageLen,uint64_t platformId,uint32_t version)73 void SocketCallbacks::handleHubInfoResponse(const char* name, const char* vendor,
74                                             const char* toolchain, uint32_t legacyPlatformVersion,
75                                             uint32_t legacyToolchainVersion, float peakMips,
76                                             float stoppedPower, float sleepPower, float peakPower,
77                                             uint32_t maxMessageLen, uint64_t platformId,
78                                             uint32_t version) {
79     LOG(VERBOSE) << "Hub Info response";
80     LOG(VERBOSE) << "Hub Info name: " << name;
81     LOG(VERBOSE) << "Version : " << version;
82     LOG(VERBOSE) << "Legacy Platform Version: " << legacyPlatformVersion;
83     LOG(VERBOSE) << "Legacy Toolchain Version: " << legacyToolchainVersion;
84     LOG(VERBOSE) << "Peak Mips: " << peakMips;
85     LOG(VERBOSE) << "Stopped Power: " << stoppedPower;
86     LOG(VERBOSE) << "Sleep Power: " << sleepPower;
87     LOG(VERBOSE) << "Peak Power: " << peakPower;
88     LOG(VERBOSE) << "Platform ID: " << platformId;
89     LOG(VERBOSE) << "Vendor : " << vendor;
90     LOG(VERBOSE) << "Toolchain : " << toolchain;
91     LOG(VERBOSE) << "maxMessageLen : " << maxMessageLen;
92     if (maxMessageLen < chre_constants::kMaxMessageLen) {
93         LOG(WARNING) << "Incorrect max message length";
94     }
95 }
96 
handleNanoappListResponse(const fbs::NanoappListResponseT & response)97 void SocketCallbacks::handleNanoappListResponse(const fbs::NanoappListResponseT& response) {
98     LOG(VERBOSE) << "handleNanoAppListResponse";
99     for (const std::unique_ptr<fbs::NanoappListEntryT>& nanoapp : response.nanoapps) {
100         if (nanoapp == nullptr) {
101             continue;
102         }
103         if (nanoapp->app_id == chre_constants::kWifiOffloadNanoAppId && nanoapp->enabled) {
104             LOG(INFO) << "Wifi Offload Nano app found";
105             LOG(INFO) << "Version: " << nanoapp->version;
106             break;
107         }
108     }
109 }
110 
handleLoadNanoappResponse(const fbs::LoadNanoappResponseT & response)111 void SocketCallbacks::handleLoadNanoappResponse(const fbs::LoadNanoappResponseT& response) {
112     LOG(VERBOSE) << "Load Nano app response";
113     LOG(VERBOSE) << "Transaction ID: " << response.transaction_id;
114     LOG(VERBOSE) << "Status: " << response.success;
115 }
116 
handleUnloadNanoappResponse(const fbs::UnloadNanoappResponseT & response)117 void SocketCallbacks::handleUnloadNanoappResponse(const fbs::UnloadNanoappResponseT& response) {
118     LOG(VERBOSE) << "Unload Nano app response";
119     LOG(VERBOSE) << "Transaction ID: " << response.transaction_id;
120     LOG(VERBOSE) << "Status: " << response.success;
121 }
122 
ChreInterface(ChreInterfaceCallbacks * callback)123 ChreInterface::ChreInterface(ChreInterfaceCallbacks* callback)
124     : mSocketCallbacks(new SocketCallbacks(this)), mServerCallbacks(callback),
125       mSocketConnected(false) {
126     if (!mClient.connectInBackground(chre_constants::kSocketName, mSocketCallbacks)) {
127         LOG(ERROR) << "Offload HAL is not connected to Chre";
128     }
129 }
130 
~ChreInterface()131 ChreInterface::~ChreInterface() {
132     mClient.disconnect();
133 }
134 
isConnected()135 bool ChreInterface::isConnected() {
136     std::lock_guard<std::mutex> lock(mChreInterfaceLock);
137     return mSocketConnected;
138 }
139 
reportConnectionEvent(ChreInterfaceCallbacks::ConnectionEvent event)140 void ChreInterface::reportConnectionEvent(ChreInterfaceCallbacks::ConnectionEvent event) {
141     bool connectionStatus = false;
142     switch (event) {
143         case ChreInterfaceCallbacks::ConnectionEvent::CONNECTED:
144             connectionStatus = true;
145             if (!getHubInfo() || !getNanoAppList()) {
146                 LOG(WARNING) << "Unable to get platform and nano app info";
147             }
148             break;
149         case ChreInterfaceCallbacks::ConnectionEvent::DISCONNECTED:
150         case ChreInterfaceCallbacks::ConnectionEvent::CONNECTION_ABORT:
151             break;
152         default:
153             LOG(WARNING) << "Invalid connection event recieved";
154             return;
155     }
156     {
157         std::lock_guard<std::mutex> lock(mChreInterfaceLock);
158         mSocketConnected = connectionStatus;
159     }
160     mServerCallbacks->handleConnectionEvents(event);
161 }
162 
sendCommandToApp(uint32_t messageType,const std::vector<uint8_t> & message)163 bool ChreInterface::sendCommandToApp(uint32_t messageType, const std::vector<uint8_t>& message) {
164     FlatBufferBuilder builder(chre_constants::kMaxMessageLen);
165     void* messageData = nullptr;
166     size_t messageDataLen = message.size();
167     if (messageDataLen > 0) {
168         messageData = (void*)message.data();
169     }
170     HostProtocolHost::encodeNanoappMessage(builder, chre_constants::kWifiOffloadNanoAppId,
171                                            messageType, 0, messageData, messageDataLen);
172     if (!mClient.sendMessage(builder.GetBufferPointer(), builder.GetSize())) {
173         LOG(WARNING) << "Failed to send message to Nano app";
174         return false;
175     }
176     LOG(VERBOSE) << "Command sent " << messageType;
177     return true;
178 }
179 
handleMessage(uint32_t messageType,const void * messageData,size_t messageDataLen)180 void ChreInterface::handleMessage(uint32_t messageType, const void* messageData,
181                                   size_t messageDataLen) {
182     const uint8_t* messageBuf = reinterpret_cast<const uint8_t*>(messageData);
183     std::vector<uint8_t> message(messageBuf, messageBuf + messageDataLen);
184     mServerCallbacks->handleMessage(messageType, message);
185 }
186 
getHubInfo()187 bool ChreInterface::getHubInfo() {
188     LOG(VERBOSE) << "getHubInfo";
189 
190     FlatBufferBuilder builder(chre_constants::kHubInfoRequestBufLen);
191     HostProtocolHost::encodeHubInfoRequest(builder);
192     if (!mClient.sendMessage(builder.GetBufferPointer(), builder.GetSize())) {
193         LOG(WARNING) << "Failed to send Hub Info request";
194         return false;
195     }
196     return true;
197 }
198 
getNanoAppList()199 bool ChreInterface::getNanoAppList() {
200     LOG(VERBOSE) << "getNanoAppList";
201     FlatBufferBuilder builder(chre_constants::kNanoAppListRequestBufLen);
202     HostProtocolHost::encodeNanoappListRequest(builder);
203 
204     if (!mClient.sendMessage(builder.GetBufferPointer(), builder.GetSize())) {
205         LOG(WARNING) << "Unable to send Nano app List request";
206         return false;
207     }
208     return true;
209 }
210 }  // namespace implementation
211 }  // namespace V1_0
212 }  // namespace offload
213 }  // namespace wifi
214 }  // namespace hardware
215 }  // namespace android
216