You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
207 lines
6.3 KiB
207 lines
6.3 KiB
/*
|
|
* Copyright (C) 2016 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#ifndef _NANOHUB_HAL_H_
|
|
#define _NANOHUB_HAL_H_
|
|
|
|
#include <mutex>
|
|
#include <thread>
|
|
#include <list>
|
|
|
|
#include <hardware/context_hub.h>
|
|
|
|
#include <nanohub/nanohub.h>
|
|
|
|
//as per protocol
|
|
#define MAX_RX_PACKET 128
|
|
#define MAX_TX_PACKET 128
|
|
#define APP_FROM_HOST_EVENT_ID 0x000000F8
|
|
#define APP_FROM_HOST_CHRE_EVENT_ID 0x000000F9
|
|
|
|
#define ENDPOINT_UNSPECIFIED 0xFFFE
|
|
#define ENDPOINT_BROADCAST 0xFFFF
|
|
|
|
namespace android {
|
|
|
|
namespace nanohub {
|
|
|
|
void dumpBuffer(const char *pfx, const hub_app_name_t &appId, uint32_t evtId, uint16_t endpoint, const void *data, size_t len, int status = 0);
|
|
|
|
struct nano_message_chre {
|
|
HostMsgHdrChre hdr;
|
|
uint8_t data[MAX_RX_PACKET];
|
|
} __attribute__((packed));
|
|
|
|
struct nano_message_raw {
|
|
HostMsgHdr hdr;
|
|
uint8_t data[MAX_RX_PACKET];
|
|
} __attribute__((packed));
|
|
|
|
union nano_message {
|
|
struct nano_message_chre chre;
|
|
struct nano_message_raw raw;
|
|
} __attribute__((packed));
|
|
|
|
class HubMessage : public hub_message_t {
|
|
std::unique_ptr<uint8_t> data_;
|
|
public:
|
|
uint32_t message_transaction_id;
|
|
uint16_t message_endpoint;
|
|
HubMessage(const HubMessage &other) = delete;
|
|
HubMessage &operator = (const HubMessage &other) = delete;
|
|
|
|
HubMessage(const hub_app_name_t *name, uint32_t typ, uint32_t transaction_id,
|
|
uint16_t endpoint, const void *data, uint32_t len) {
|
|
app_name = *name;
|
|
message_type = typ;
|
|
message_len = len;
|
|
message = data;
|
|
message_transaction_id = transaction_id;
|
|
message_endpoint = endpoint;
|
|
if (len > 0 && data != nullptr) {
|
|
data_ = std::unique_ptr<uint8_t>(new uint8_t[len]);
|
|
memcpy(data_.get(), data, len);
|
|
message = data_.get();
|
|
}
|
|
}
|
|
|
|
HubMessage(const hub_app_name_t *name, uint32_t typ, uint16_t endpoint, const void *data,
|
|
uint32_t len) : HubMessage(name, typ, 0, endpoint, data, len) { }
|
|
|
|
HubMessage(const hub_message_t *msg, uint32_t transaction_id, uint16_t endpoint) {
|
|
app_name = msg->app_name;
|
|
message_type = msg->message_type;
|
|
message_len = msg->message_len;
|
|
message = msg->message;
|
|
message_transaction_id = transaction_id;
|
|
message_endpoint = endpoint;
|
|
if (msg->message_len > 0 && msg->message != nullptr) {
|
|
data_ = std::unique_ptr<uint8_t>(new uint8_t[msg->message_len]);
|
|
memcpy(data_.get(), msg->message, msg->message_len);
|
|
message = data_.get();
|
|
}
|
|
}
|
|
|
|
HubMessage(HubMessage &&other) {
|
|
*this = (HubMessage &&)other;
|
|
}
|
|
|
|
HubMessage &operator = (HubMessage &&other) {
|
|
*static_cast<hub_message_t *>(this) = static_cast<hub_message_t>(other);
|
|
message_transaction_id = other.message_transaction_id;
|
|
message_endpoint = other.message_endpoint;
|
|
data_ = std::move(other.data_);
|
|
other.message = nullptr;
|
|
other.message_len = 0;
|
|
return *this;
|
|
}
|
|
};
|
|
|
|
typedef int Contexthub_callback(uint32_t hub_id, const HubMessage &rxed_msg, void *cookie);
|
|
|
|
class NanoHub {
|
|
std::mutex mLock;
|
|
bool mAppQuit;
|
|
std::mutex mAppTxLock;
|
|
std::condition_variable mAppTxCond;
|
|
std::list<HubMessage> mAppTxQueue;
|
|
std::thread mPollThread;
|
|
std::thread mAppThread;
|
|
Contexthub_callback *mMsgCbkFunc;
|
|
int mThreadClosingPipe[2];
|
|
int mFd; // [0] is read end
|
|
void * mMsgCbkData;
|
|
|
|
NanoHub();
|
|
~NanoHub();
|
|
|
|
void reset() {
|
|
mThreadClosingPipe[0] = -1;
|
|
mThreadClosingPipe[1] = -1;
|
|
mFd = -1;
|
|
mMsgCbkData = nullptr;
|
|
mMsgCbkFunc = nullptr;
|
|
mAppQuit = false;
|
|
}
|
|
|
|
void* runAppTx();
|
|
void* runDeviceRx();
|
|
|
|
int openHub();
|
|
int closeHub();
|
|
|
|
static NanoHub *hubInstance() {
|
|
static NanoHub theHub;
|
|
return &theHub;
|
|
}
|
|
|
|
int doSubscribeMessages(uint32_t hub_id, Contexthub_callback *cbk, void *cookie);
|
|
int doSendToNanohub(uint32_t hub_id, const hub_message_t *msg,
|
|
uint32_t transaction_id, uint16_t endpoint);
|
|
int doSendToDevice(const hub_app_name_t name, const void *data, uint32_t len,
|
|
uint32_t messageType = 0, uint16_t endpoint = ENDPOINT_UNSPECIFIED);
|
|
void doSendToApp(HubMessage &&msg);
|
|
void doDumpAppInfo(std::string &result);
|
|
|
|
static constexpr unsigned int FL_MESSAGE_TRACING = 1;
|
|
|
|
unsigned int mFlags = 0;
|
|
|
|
public:
|
|
|
|
// debugging interface
|
|
|
|
static bool messageTracingEnabled() {
|
|
return hubInstance()->mFlags & FL_MESSAGE_TRACING;
|
|
}
|
|
static unsigned int getDebugFlags() {
|
|
return hubInstance()->mFlags;
|
|
}
|
|
static void setDebugFlags(unsigned int flags) {
|
|
hubInstance()->mFlags = flags;
|
|
}
|
|
static void dumpAppInfo(std::string &result) {
|
|
hubInstance()->doDumpAppInfo(result);
|
|
}
|
|
|
|
// messaging interface
|
|
|
|
// define callback to invoke for APP messages
|
|
static int subscribeMessages(uint32_t hub_id, Contexthub_callback *cbk, void *cookie) {
|
|
return hubInstance()->doSubscribeMessages(hub_id, cbk, cookie);
|
|
}
|
|
// all messages from APP go here
|
|
static int sendToNanohub(uint32_t hub_id, const hub_message_t *msg,
|
|
uint32_t transaction_id, uint16_t endpoint) {
|
|
return hubInstance()->doSendToNanohub(hub_id, msg, transaction_id, endpoint);
|
|
}
|
|
// passes message to kernel driver directly
|
|
static int sendToDevice(const hub_app_name_t *name, const void *data, uint32_t len,
|
|
uint32_t transactionId) {
|
|
return hubInstance()->doSendToDevice(*name, data, len, transactionId, ENDPOINT_UNSPECIFIED);
|
|
}
|
|
// passes message to APP via callback
|
|
static void sendToApp(HubMessage &&msg) {
|
|
hubInstance()->doSendToApp((HubMessage &&)msg);
|
|
}
|
|
};
|
|
|
|
}; // namespace nanohub
|
|
|
|
}; // namespace android
|
|
|
|
#endif
|