| /* |
| * ANT Stack |
| * |
| * Copyright 2011 Dynastream Innovations |
| * |
| * 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. |
| */ |
| /******************************************************************************\ |
| * |
| * FILE NAME: ant_rx_chardev.c |
| * |
| * BRIEF: |
| * This file implements the receive thread function which will loop reading |
| * ANT messages until told to exit. |
| * |
| * |
| \******************************************************************************/ |
| |
| #include <errno.h> |
| #include <poll.h> |
| #include <pthread.h> |
| #include <stdint.h> /* for uint64_t */ |
| #include <string.h> |
| |
| #include "ant_types.h" |
| #include "antradio_power.h" |
| #include "ant_rx_chardev.h" |
| #include "ant_hci_defines.h" |
| #include "ant_log.h" |
| #include "ant_native.h" // ANT_HCI_MAX_MSG_SIZE, ANT_MSG_ID_OFFSET, ANT_MSG_DATA_OFFSET, |
| // ant_radio_enabled_status() |
| |
| #include "AntHidlClient.h" |
| extern ANTStatus ant_tx_message_flowcontrol_none(ant_channel_type eTxPath, ANT_U8 ucMessageLength, ANT_U8 *pucTxMessage); |
| |
| #undef LOG_TAG |
| #define LOG_TAG "antradio_rx" |
| |
| #define ANT_POLL_TIMEOUT ((int)30000) |
| #define KEEPALIVE_TIMEOUT ((int)5000) |
| |
| ANT_U8 aucRxBuffer[NUM_ANT_CHANNELS][ANT_HCI_MAX_MSG_SIZE]; |
| |
| #ifdef ANT_DEVICE_NAME // Single transport path |
| int iRxBufferLength[NUM_ANT_CHANNELS] = {0}; |
| #else |
| int iRxBufferLength[NUM_ANT_CHANNELS] = {0, 0}; |
| #endif // |
| |
| // Plus one is for the eventfd shutdown signal. |
| #define NUM_POLL_FDS (NUM_ANT_CHANNELS + 1) |
| #define EVENTFD_IDX NUM_ANT_CHANNELS |
| |
| static ANT_U8 KEEPALIVE_MESG[] = {0x01, 0x00, 0x00}; |
| static ANT_U8 KEEPALIVE_RESP[] = {0x03, 0x40, 0x00, 0x00, 0x28}; |
| |
| void doReset(ant_rx_thread_info_t *stRxThreadInfo); |
| int readChannelMsg(ant_channel_type eChannel, ant_channel_info_t *pstChnlInfo); |
| |
| |
| /* |
| * This thread is run occasionally as a detached thread in order to send a keepalive message to the |
| * chip. |
| */ |
| void *fnKeepAliveThread(void *unused) |
| { |
| ANT_DEBUG_V("Sending dummy keepalive message."); |
| ant_tx_message(sizeof(KEEPALIVE_MESG)/sizeof(ANT_U8), KEEPALIVE_MESG); |
| return NULL; |
| } |
| |
| /* |
| * This thread waits for ANT messages from a VFS file. |
| */ |
| void *fnRxThread(void *ant_rx_thread_info) |
| { |
| int iMutexLockResult; |
| int iPollRet; |
| ant_rx_thread_info_t *stRxThreadInfo; |
| struct pollfd astPollFd[NUM_POLL_FDS]; |
| ANT_FUNC_START(); |
| |
| stRxThreadInfo = (ant_rx_thread_info_t *)ant_rx_thread_info; |
| // Fill out poll request for the shutdown signaller. |
| astPollFd[EVENTFD_IDX].fd = stRxThreadInfo->iRxShutdownEventFd; |
| astPollFd[EVENTFD_IDX].events = POLL_IN; |
| |
| // Reset the waiting for response, since we don't want a stale value if we were reset. |
| stRxThreadInfo->bWaitingForKeepaliveResponse = ANT_FALSE; |
| |
| /* continue running as long as not terminated */ |
| while (stRxThreadInfo->ucRunThread) { |
| iPollRet = ant_rx_check(); |
| if(iPollRet == 0) |
| { |
| readChannelMsg(0, &stRxThreadInfo->astChannels[0]); |
| } |
| } |
| |
| /* disable ANT radio if not already disabling */ |
| // Try to get stEnabledStatusLock. |
| // if you get it then no one is enabling or disabling |
| // if you can't get it assume something made you exit |
| ANT_DEBUG_V("try getting stEnabledStatusLock in %s", __FUNCTION__); |
| iMutexLockResult = pthread_mutex_trylock(stRxThreadInfo->pstEnabledStatusLock); |
| if (!iMutexLockResult) { |
| ANT_DEBUG_V("got stEnabledStatusLock in %s", __FUNCTION__); |
| ANT_WARN("rx thread has unexpectedly crashed, cleaning up"); |
| |
| // spoof our handle as closed so we don't try to join ourselves in disable |
| stRxThreadInfo->stRxThread = 0; |
| |
| if (g_fnStateCallback) { |
| g_fnStateCallback(RADIO_STATUS_DISABLING); |
| } |
| |
| ant_disable(); |
| |
| if (g_fnStateCallback) { |
| g_fnStateCallback(ant_radio_enabled_status()); |
| } |
| |
| ANT_DEBUG_V("releasing stEnabledStatusLock in %s", __FUNCTION__); |
| pthread_mutex_unlock(stRxThreadInfo->pstEnabledStatusLock); |
| ANT_DEBUG_V("released stEnabledStatusLock in %s", __FUNCTION__); |
| } else if (iMutexLockResult != EBUSY) { |
| ANT_ERROR("rx thread closing code, trylock on state lock failed: %s", |
| strerror(iMutexLockResult)); |
| } else { |
| ANT_DEBUG_V("stEnabledStatusLock busy"); |
| } |
| |
| out: |
| ANT_FUNC_END(); |
| #ifdef ANDROID |
| return NULL; |
| #endif |
| } |
| |
| void doReset(ant_rx_thread_info_t *stRxThreadInfo) |
| { |
| int iMutexLockResult; |
| |
| ANT_FUNC_START(); |
| /* Chip was reset or other error, only way to recover is to |
| * close and open ANT chardev */ |
| stRxThreadInfo->ucChipResetting = 1; |
| |
| if (g_fnStateCallback) { |
| g_fnStateCallback(RADIO_STATUS_RESETTING); |
| } |
| |
| stRxThreadInfo->ucRunThread = 0; |
| |
| ANT_DEBUG_V("getting stEnabledStatusLock in %s", __FUNCTION__); |
| iMutexLockResult = pthread_mutex_lock(stRxThreadInfo->pstEnabledStatusLock); |
| if (iMutexLockResult < 0) { |
| ANT_ERROR("chip was reset, getting state mutex failed: %s", |
| strerror(iMutexLockResult)); |
| stRxThreadInfo->stRxThread = 0; |
| stRxThreadInfo->ucChipResetting = 0; |
| } else { |
| ANT_DEBUG_V("got stEnabledStatusLock in %s", __FUNCTION__); |
| |
| stRxThreadInfo->stRxThread = 0; /* spoof our handle as closed so we don't |
| * try to join ourselves in disable */ |
| |
| ant_disable(); |
| |
| int enableResult = ant_enable(); |
| |
| stRxThreadInfo->ucChipResetting = 0; |
| if (enableResult) { /* failed */ |
| if (g_fnStateCallback) { |
| g_fnStateCallback(RADIO_STATUS_DISABLED); |
| } |
| } else { /* success */ |
| if (g_fnStateCallback) { |
| g_fnStateCallback(RADIO_STATUS_RESET); |
| } |
| } |
| |
| ANT_DEBUG_V("releasing stEnabledStatusLock in %s", __FUNCTION__); |
| pthread_mutex_unlock(stRxThreadInfo->pstEnabledStatusLock); |
| ANT_DEBUG_V("released stEnabledStatusLock in %s", __FUNCTION__); |
| } |
| |
| ANT_FUNC_END(); |
| } |
| |
| //////////////////////////////////////////////////////////////////// |
| // setFlowControl |
| // |
| // Sets the flow control "flag" to the value provided and signals the transmit |
| // thread to check the value. |
| // |
| // Parameters: |
| // pstChnlInfo the details of the channel being updated |
| // ucFlowSetting the value to use |
| // |
| // Returns: |
| // Success: |
| // 0 |
| // Failure: |
| // -1 |
| //////////////////////////////////////////////////////////////////// |
| int setFlowControl(ant_channel_info_t *pstChnlInfo, ANT_U8 ucFlowSetting) |
| { |
| int iRet = -1; |
| int iMutexResult; |
| ANT_FUNC_START(); |
| |
| ANT_DEBUG_V("getting stFlowControlLock in %s", __FUNCTION__); |
| iMutexResult = pthread_mutex_lock(pstChnlInfo->pstFlowControlLock); |
| if (iMutexResult) { |
| ANT_ERROR("failed to lock flow control mutex during response: %s", strerror(iMutexResult)); |
| } else { |
| ANT_DEBUG_V("got stFlowControlLock in %s", __FUNCTION__); |
| |
| pstChnlInfo->ucFlowControlResp = ucFlowSetting; |
| |
| ANT_DEBUG_V("releasing stFlowControlLock in %s", __FUNCTION__); |
| pthread_mutex_unlock(pstChnlInfo->pstFlowControlLock); |
| ANT_DEBUG_V("released stFlowControlLock in %s", __FUNCTION__); |
| |
| pthread_cond_signal(pstChnlInfo->pstFlowControlCond); |
| |
| iRet = 0; |
| } |
| |
| ANT_FUNC_END(); |
| return iRet; |
| } |
| |
| int readChannelMsg(ant_channel_type eChannel, ant_channel_info_t *pstChnlInfo) |
| { |
| int iRet = -1; |
| int iRxLenRead = 0; |
| int iCurrentHciPacketOffset; |
| int iHciDataSize; |
| ANT_FUNC_START(); |
| |
| // Keep trying to read while there is an error, and that error is EAGAIN |
| { |
| iRxLenRead += iRxBufferLength[eChannel]; // add existing data on |
| ANT_DEBUG_D("iRxLenRead %d",iRxLenRead); |
| ANT_SERIAL(aucRxBuffer[eChannel], iRxLenRead, 'R'); |
| |
| // if we didn't get a full packet, then just exit |
| if (iRxLenRead < (aucRxBuffer[eChannel][ANT_HCI_SIZE_OFFSET] + ANT_HCI_HEADER_SIZE + ANT_HCI_FOOTER_SIZE)) { |
| iRxBufferLength[eChannel] = iRxLenRead; |
| iRet = 0; |
| goto out; |
| } |
| |
| iRxBufferLength[eChannel] = 0; // reset buffer length here since we should have a full packet |
| |
| #if ANT_HCI_OPCODE_SIZE == 1 // Check the different message types by opcode |
| ANT_U8 opcode = aucRxBuffer[eChannel][ANT_HCI_OPCODE_OFFSET]; |
| |
| if(ANT_HCI_OPCODE_COMMAND_COMPLETE == opcode) { |
| // Command Complete, so signal a FLOW_GO |
| if(setFlowControl(pstChnlInfo, ANT_FLOW_GO)) { |
| goto out; |
| } |
| } else if(ANT_HCI_OPCODE_FLOW_ON == opcode) { |
| // FLow On, so resend the last Tx |
| #ifdef ANT_FLOW_RESEND |
| // Check if there is a message to resend |
| if(pstChnlInfo->ucResendMessageLength > 0) { |
| ant_tx_message_flowcontrol_none(eChannel, pstChnlInfo->ucResendMessageLength, pstChnlInfo->pucResendMessage); |
| } else { |
| ANT_DEBUG_D("Resend requested by chip, but tx request cancelled"); |
| } |
| #endif // ANT_FLOW_RESEND |
| } else if(ANT_HCI_OPCODE_ANT_EVENT == opcode) |
| // ANT Event, send ANT packet to Rx Callback |
| #endif // ANT_HCI_OPCODE_SIZE == 1 |
| { |
| // Received an ANT packet |
| iCurrentHciPacketOffset = 0; |
| |
| while(iCurrentHciPacketOffset < iRxLenRead) { |
| ANT_DEBUG_D("iRxLenRead = %d",iRxLenRead); |
| |
| // TODO Allow HCI Packet Size value to be larger than 1 byte |
| // This currently works as no size value is greater than 255, and little endian |
| iHciDataSize = aucRxBuffer[eChannel][iCurrentHciPacketOffset + ANT_HCI_SIZE_OFFSET]; |
| |
| if ((iHciDataSize + ANT_HCI_HEADER_SIZE + ANT_HCI_FOOTER_SIZE + iCurrentHciPacketOffset) > |
| iRxLenRead) { |
| // we don't have a whole packet |
| iRxBufferLength[eChannel] = iRxLenRead - iCurrentHciPacketOffset; |
| memcpy(aucRxBuffer[eChannel], &aucRxBuffer[eChannel][iCurrentHciPacketOffset], iRxBufferLength[eChannel]); |
| // the increment at the end should push us out of the while loop |
| } else |
| #ifdef ANT_MESG_FLOW_CONTROL |
| if (aucRxBuffer[eChannel][iCurrentHciPacketOffset + ANT_HCI_DATA_OFFSET + ANT_MSG_ID_OFFSET] == |
| ANT_MESG_FLOW_CONTROL) { |
| // This is a flow control packet, not a standard ANT message |
| if(setFlowControl(pstChnlInfo, \ |
| aucRxBuffer[eChannel][iCurrentHciPacketOffset + ANT_HCI_DATA_OFFSET + ANT_MSG_DATA_OFFSET])) { |
| goto out; |
| } |
| } else |
| #endif // ANT_MESG_FLOW_CONTROL |
| { |
| ANT_U8 *msg = aucRxBuffer[eChannel] + iCurrentHciPacketOffset + ANT_HCI_DATA_OFFSET; |
| ANT_BOOL bIsKeepAliveResponse = memcmp(msg, KEEPALIVE_RESP, sizeof(KEEPALIVE_RESP)/sizeof(ANT_U8)) == 0; |
| if (bIsKeepAliveResponse) { |
| ANT_DEBUG_V("Filtered out keepalive response."); |
| } else if (pstChnlInfo->fnRxCallback != NULL) { |
| ANT_DEBUG_V("call rx callback hci data size = %d",iHciDataSize); |
| |
| // Loop through read data until all HCI packets are written to callback |
| pstChnlInfo->fnRxCallback(iHciDataSize, \ |
| msg); |
| } else { |
| ANT_WARN("%s rx callback is null", pstChnlInfo->pcDevicePath); |
| } |
| } |
| |
| iCurrentHciPacketOffset = iCurrentHciPacketOffset + ANT_HCI_HEADER_SIZE + ANT_HCI_FOOTER_SIZE + iHciDataSize; |
| } |
| } |
| |
| iRet = 0; |
| } |
| |
| out: |
| // Need to indicate that we are done handling the rx buffer and it can be |
| // overwritten again. |
| ant_rx_clear(); |
| ANT_FUNC_END(); |
| return iRet; |
| } |