|
|
|
/*
|
|
|
|
LUFA Library
|
|
|
|
Copyright (C) Dean Camera, 2010.
|
|
|
|
|
|
|
|
dean [at] fourwalledcubicle [dot] com
|
|
|
|
www.fourwalledcubicle.com
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
Copyright 2010 Dean Camera (dean [at] fourwalledcubicle [dot] com)
|
|
|
|
|
|
|
|
Permission to use, copy, modify, distribute, and sell this
|
|
|
|
software and its documentation for any purpose is hereby granted
|
|
|
|
without fee, provided that the above copyright notice appear in
|
|
|
|
all copies and that both that the copyright notice and this
|
|
|
|
permission notice and warranty disclaimer appear in supporting
|
|
|
|
documentation, and that the name of the author not be used in
|
|
|
|
advertising or publicity pertaining to distribution of the
|
|
|
|
software without specific, written prior permission.
|
|
|
|
|
|
|
|
The author disclaim all warranties with regard to this
|
|
|
|
software, including all implied warranties of merchantability
|
|
|
|
and fitness. In no event shall the author be liable for any
|
|
|
|
special, indirect or consequential damages or any damages
|
|
|
|
whatsoever resulting from loss of use, data or profits, whether
|
|
|
|
in an action of contract, negligence or other tortious action,
|
|
|
|
arising out of or in connection with the use or performance of
|
|
|
|
this software.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define INCLUDE_FROM_BLUETOOTHHCICOMMANDS_C
|
|
|
|
#include "BluetoothHCICommands.h"
|
|
|
|
|
|
|
|
/** Current processing state of the HCI portion of the Bluetooth stack. */
|
|
|
|
uint8_t Bluetooth_HCIProcessingState;
|
|
|
|
|
|
|
|
/** Next HCI state to enter once the last issued HCI command has completed. */
|
|
|
|
static uint8_t Bluetooth_HCINextState;
|
|
|
|
|
|
|
|
/** Temporary Bluetooth Device Address, for HCI responses which much include the detination address */
|
|
|
|
static uint8_t Bluetooth_TempDeviceAddress[6];
|
|
|
|
|
|
|
|
|
|
|
|
/** Bluetooth HCI processing task. This task should be called repeatedly the main Bluetooth
|
|
|
|
* stack task to manage the HCI processing state.
|
|
|
|
*/
|
|
|
|
void Bluetooth_HCITask(void)
|
|
|
|
{
|
|
|
|
BT_HCICommand_Header_t HCICommandHeader;
|
|
|
|
|
|
|
|
switch (Bluetooth_HCIProcessingState)
|
|
|
|
{
|
|
|
|
case Bluetooth_ProcessEvents:
|
|
|
|
Pipe_SelectPipe(BLUETOOTH_EVENTS_PIPE);
|
|
|
|
Pipe_Unfreeze();
|
|
|
|
|
|
|
|
if (Pipe_IsReadWriteAllowed())
|
|
|
|
{
|
|
|
|
BT_HCIEvent_Header_t HCIEventHeader;
|
|
|
|
|
|
|
|
/* Read in the event header to fetch the event code and payload length */
|
|
|
|
Pipe_Read_Stream_LE(&HCIEventHeader, sizeof(HCIEventHeader));
|
|
|
|
|
|
|
|
/* Create a temporary buffer for the event parameters */
|
|
|
|
uint8_t EventParams[HCIEventHeader.ParameterLength];
|
|
|
|
|
|
|
|
/* Read in the event parameters into the temporary buffer */
|
|
|
|
Pipe_Read_Stream_LE(&EventParams, HCIEventHeader.ParameterLength);
|
|
|
|
Pipe_ClearIN();
|
|
|
|
|
|
|
|
BT_HCI_DEBUG(1, "Event Received (0x%02X)", HCIEventHeader.EventCode);
|
|
|
|
|
|
|
|
switch (HCIEventHeader.EventCode)
|
|
|
|
{
|
|
|
|
case EVENT_COMMAND_COMPLETE:
|
|
|
|
BT_HCI_DEBUG(1, "<< Command Complete", NULL);
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_HCINextState;
|
|
|
|
break;
|
|
|
|
case EVENT_COMMAND_STATUS:
|
|
|
|
BT_HCI_DEBUG(1, "<< Command Status", NULL);
|
|
|
|
BT_HCI_DEBUG(2, "-- Status Code: 0x%02X", (((BT_HCIEvent_CommandStatus_t*)&EventParams)->Status));
|
|
|
|
|
|
|
|
/* If the execution of a command failed, reset the stack */
|
|
|
|
if (((BT_HCIEvent_CommandStatus_t*)&EventParams)->Status)
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_Init;
|
|
|
|
break;
|
|
|
|
case EVENT_CONNECTION_REQUEST:
|
|
|
|
BT_HCI_DEBUG(1, "<< Connection Request", NULL);
|
|
|
|
BT_HCI_DEBUG(2, "-- Link Type: 0x%02X", (((BT_HCIEvent_ConnectionRequest_t*)&EventParams)->LinkType));
|
|
|
|
|
|
|
|
/* Need to store the remote device's BT address in a temporary buffer for later use */
|
|
|
|
memcpy(Bluetooth_TempDeviceAddress,
|
|
|
|
&((BT_HCIEvent_ConnectionRequest_t*)&EventParams)->RemoteAddress,
|
|
|
|
sizeof(Bluetooth_TempDeviceAddress));
|
|
|
|
|
|
|
|
bool IsACLConnection = (((BT_HCIEvent_ConnectionRequest_t*)&EventParams)->LinkType == 0x01);
|
|
|
|
|
|
|
|
/* Only accept the connection if it is a ACL (data) connection, a device is not already connected
|
|
|
|
and the user application has indicated that the connection should be allowed */
|
|
|
|
Bluetooth_HCIProcessingState = (Bluetooth_Connection.IsConnected || !(IsACLConnection) ||
|
|
|
|
!(Bluetooth_ConnectionRequest(Bluetooth_TempDeviceAddress))) ?
|
|
|
|
Bluetooth_Conn_RejectConnection : Bluetooth_Conn_AcceptConnection;
|
|
|
|
|
|
|
|
BT_HCI_DEBUG(2, "-- Connection %S", (Bluetooth_HCIProcessingState == Bluetooth_Conn_RejectConnection) ?
|
|
|
|
PSTR("REJECTED") : PSTR("ACCEPTED"));
|
|
|
|
|
|
|
|
break;
|
|
|
|
case EVENT_PIN_CODE_REQUEST:
|
|
|
|
BT_HCI_DEBUG(1, "<< Pin Code Request", NULL);
|
|
|
|
|
|
|
|
/* Need to store the remote device's BT address in a temporary buffer for later use */
|
|
|
|
memcpy(Bluetooth_TempDeviceAddress,
|
|
|
|
&((BT_HCIEvent_PinCodeReq_t*)&EventParams)->RemoteAddress,
|
|
|
|
sizeof(Bluetooth_TempDeviceAddress));
|
|
|
|
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_Conn_SendPINCode;
|
|
|
|
break;
|
|
|
|
case EVENT_LINK_KEY_REQUEST:
|
|
|
|
BT_HCI_DEBUG(1, "<< Link Key Request", NULL);
|
|
|
|
|
|
|
|
/* Need to store the remote device's BT address in a temporary buffer for later use */
|
|
|
|
memcpy(Bluetooth_TempDeviceAddress,
|
|
|
|
&((BT_HCIEvent_LinkKeyReq_t*)&EventParams)->RemoteAddress,
|
|
|
|
sizeof(Bluetooth_TempDeviceAddress));
|
|
|
|
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_Conn_SendLinkKeyNAK;
|
|
|
|
break;
|
|
|
|
case EVENT_CONNECTION_COMPLETE:
|
|
|
|
BT_HCI_DEBUG(1, "<< Connection Complete", NULL);
|
|
|
|
BT_HCI_DEBUG(2, "-- Handle: 0x%04X", ((BT_HCIEvent_ConnectionComplete_t*)&EventParams)->ConnectionHandle);
|
|
|
|
|
|
|
|
/* Need to store the remote device's BT address in a temporary buffer for later use */
|
|
|
|
memcpy(Bluetooth_Connection.RemoteAddress,
|
|
|
|
&((BT_HCIEvent_ConnectionComplete_t*)&EventParams)->RemoteAddress,
|
|
|
|
sizeof(Bluetooth_TempDeviceAddress));
|
|
|
|
|
|
|
|
/* Store the created connection handle and indicate that the connection has been established */
|
|
|
|
Bluetooth_Connection.ConnectionHandle = ((BT_HCIEvent_ConnectionComplete_t*)&EventParams)->ConnectionHandle;
|
|
|
|
Bluetooth_Connection.IsConnected = true;
|
|
|
|
|
|
|
|
Bluetooth_ConnectionComplete();
|
|
|
|
break;
|
|
|
|
case EVENT_DISCONNECTION_COMPLETE:
|
|
|
|
BT_HCI_DEBUG(1, "<< Disconnection Complete", NULL);
|
|
|
|
|
|
|
|
/* Device disconnected, indicate connection information no longer valid */
|
|
|
|
Bluetooth_Connection.IsConnected = false;
|
|
|
|
|
|
|
|
Bluetooth_DisconnectionComplete();
|
|
|
|
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_Init;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Pipe_Freeze();
|
|
|
|
|
|
|
|
break;
|
|
|
|
case Bluetooth_Init:
|
|
|
|
BT_HCI_DEBUG(1, "# Init", NULL);
|
|
|
|
|
|
|
|
/* Reset the connection information structure to destroy any previous connection state */
|
|
|
|
memset(&Bluetooth_Connection, 0x00, sizeof(Bluetooth_Connection));
|
|
|
|
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_Init_Reset;
|
|
|
|
break;
|
|
|
|
case Bluetooth_Init_Reset:
|
|
|
|
BT_HCI_DEBUG(1, "# Reset", NULL);
|
|
|
|
|
|
|
|
HCICommandHeader = (BT_HCICommand_Header_t)
|
|
|
|
{
|
|
|
|
OpCode: {OGF: OGF_CTRLR_BASEBAND, OCF: OCF_CTRLR_BASEBAND_RESET},
|
|
|
|
ParameterLength: 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Send the command to reset the bluetooth dongle controller */
|
|
|
|
Bluetooth_SendHCICommand(&HCICommandHeader, NULL, 0);
|
|
|
|
|
|
|
|
Bluetooth_HCINextState = Bluetooth_Init_SetLocalName;
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_ProcessEvents;
|
|
|
|
break;
|
|
|
|
case Bluetooth_Init_SetLocalName:
|
|
|
|
BT_HCI_DEBUG(1, "# Set Local Name", NULL);
|
|
|
|
|
|
|
|
HCICommandHeader = (BT_HCICommand_Header_t)
|
|
|
|
{
|
|
|
|
OpCode: {OGF: OGF_CTRLR_BASEBAND, OCF: OCF_CTRLR_BASEBAND_WRITE_LOCAL_NAME},
|
|
|
|
ParameterLength: 248,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Send the command to set the bluetooth dongle's name for other devices to see */
|
|
|
|
Bluetooth_SendHCICommand(&HCICommandHeader, Bluetooth_DeviceConfiguration.Name, strlen(Bluetooth_DeviceConfiguration.Name));
|
|
|
|
|
|
|
|
Bluetooth_HCINextState = Bluetooth_Init_SetDeviceClass;
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_ProcessEvents;
|
|
|
|
break;
|
|
|
|
case Bluetooth_Init_SetDeviceClass:
|
|
|
|
BT_HCI_DEBUG(1, "# Set Device Class", NULL);
|
|
|
|
|
|
|
|
HCICommandHeader = (BT_HCICommand_Header_t)
|
|
|
|
{
|
|
|
|
OpCode: {OGF: OGF_CTRLR_BASEBAND, OCF: OCF_CTRLR_BASEBAND_WRITE_CLASS_OF_DEVICE},
|
|
|
|
ParameterLength: 3,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Send the command to set the class of the device for other devices to see */
|
|
|
|
Bluetooth_SendHCICommand(&HCICommandHeader, &Bluetooth_DeviceConfiguration.Class, 3);
|
|
|
|
|
|
|
|
Bluetooth_HCINextState = Bluetooth_Init_WriteScanEnable;
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_ProcessEvents;
|
|
|
|
break;
|
|
|
|
case Bluetooth_Init_WriteScanEnable:
|
|
|
|
BT_HCI_DEBUG(1, "# Write Scan Enable", NULL);
|
|
|
|
|
|
|
|
HCICommandHeader = (BT_HCICommand_Header_t)
|
|
|
|
{
|
|
|
|
OpCode: {OGF: OGF_CTRLR_BASEBAND, OCF: OCF_CTRLR_BASEBAND_WRITE_SCAN_ENABLE},
|
|
|
|
ParameterLength: 1,
|
|
|
|
};
|
|
|
|
|
|
|
|
uint8_t Interval = BT_SCANMODE_InquiryAndPageScans;
|
|
|
|
|
|
|
|
/* Send the command to set the remote device scanning mode */
|
|
|
|
Bluetooth_SendHCICommand(&HCICommandHeader, &Interval, 1);
|
|
|
|
|
|
|
|
Bluetooth_HCINextState = Bluetooth_ProcessEvents;
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_ProcessEvents;
|
|
|
|
break;
|
|
|
|
case Bluetooth_Conn_AcceptConnection:
|
|
|
|
BT_HCI_DEBUG(1, "# Accept Connection", NULL);
|
|
|
|
|
|
|
|
HCICommandHeader = (BT_HCICommand_Header_t)
|
|
|
|
{
|
|
|
|
OpCode: {OGF: OGF_LINK_CONTROL, OCF: OCF_LINK_CONTROL_ACCEPT_CONNECTION_REQUEST},
|
|
|
|
ParameterLength: sizeof(BT_HCICommand_AcceptConnectionReq_t),
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Copy over the temporary BT device address saved from the Connection Request event, indicate slave
|
|
|
|
connection role */
|
|
|
|
BT_HCICommand_AcceptConnectionReq_t AcceptConnectionParams;
|
|
|
|
memcpy(AcceptConnectionParams.RemoteAddress, Bluetooth_TempDeviceAddress,
|
|
|
|
sizeof(AcceptConnectionParams.RemoteAddress));
|
|
|
|
AcceptConnectionParams.SlaveRole = true;
|
|
|
|
|
|
|
|
/* Send the command to accept the remote connection request */
|
|
|
|
Bluetooth_SendHCICommand(&HCICommandHeader, &AcceptConnectionParams, sizeof(BT_HCICommand_AcceptConnectionReq_t));
|
|
|
|
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_ProcessEvents;
|
|
|
|
break;
|
|
|
|
case Bluetooth_Conn_RejectConnection:
|
|
|
|
BT_HCI_DEBUG(1, "# Reject Connection", NULL);
|
|
|
|
|
|
|
|
HCICommandHeader = (BT_HCICommand_Header_t)
|
|
|
|
{
|
|
|
|
OpCode: {OGF: OGF_LINK_CONTROL, OCF: OCF_LINK_CONTROL_REJECT_CONNECTION_REQUEST},
|
|
|
|
ParameterLength: sizeof(BT_HCICommand_RejectConnectionReq_t),
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Copy over the temporary BT device address saved from the Connection Request event, indicate failure
|
|
|
|
to accept the connection due to limited device resources or incorrect device address */
|
|
|
|
BT_HCICommand_RejectConnectionReq_t RejectConnectionParams;
|
|
|
|
memcpy(RejectConnectionParams.RemoteAddress, Bluetooth_TempDeviceAddress, sizeof(RejectConnectionParams.RemoteAddress));
|
|
|
|
RejectConnectionParams.Reason = Bluetooth_Connection.IsConnected ? ERROR_LIMITED_RESOURCES : ERROR_UNACCEPTABLE_BDADDR;
|
|
|
|
|
|
|
|
/* Send the command to reject the remote connection request */
|
|
|
|
Bluetooth_SendHCICommand(&HCICommandHeader, &RejectConnectionParams, sizeof(BT_HCICommand_RejectConnectionReq_t));
|
|
|
|
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_ProcessEvents;
|
|
|
|
break;
|
|
|
|
case Bluetooth_Conn_SendPINCode:
|
|
|
|
BT_HCI_DEBUG(1, "# Send Pin Code", NULL);
|
|
|
|
|
|
|
|
HCICommandHeader = (BT_HCICommand_Header_t)
|
|
|
|
{
|
|
|
|
OpCode: {OGF: OGF_LINK_CONTROL, OCF: OCF_LINK_CONTROL_PIN_CODE_REQUEST_REPLY},
|
|
|
|
ParameterLength: sizeof(BT_HCICommand_PinCodeResp_t),
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Copy over the temporary BT device address saved from the PIN Code Request event, copy over the
|
|
|
|
local PIN authentication code to the response */
|
|
|
|
BT_HCICommand_PinCodeResp_t PINCodeRequestParams;
|
|
|
|
memcpy(PINCodeRequestParams.RemoteAddress, Bluetooth_TempDeviceAddress, sizeof(PINCodeRequestParams.RemoteAddress));
|
|
|
|
PINCodeRequestParams.PINCodeLength = strlen(Bluetooth_DeviceConfiguration.PINCode);
|
|
|
|
memcpy(PINCodeRequestParams.PINCode, Bluetooth_DeviceConfiguration.PINCode, sizeof(PINCodeRequestParams.PINCode));
|
|
|
|
|
|
|
|
/* Send the command to transmit the device's local PIN number for authentication */
|
|
|
|
Bluetooth_SendHCICommand(&HCICommandHeader, &PINCodeRequestParams, sizeof(BT_HCICommand_PinCodeResp_t));
|
|
|
|
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_ProcessEvents;
|
|
|
|
break;
|
|
|
|
case Bluetooth_Conn_SendLinkKeyNAK:
|
|
|
|
BT_HCI_DEBUG(1, "# Send Link Key NAK", NULL);
|
|
|
|
|
|
|
|
HCICommandHeader = (BT_HCICommand_Header_t)
|
|
|
|
{
|
|
|
|
OpCode: {OGF: OGF_LINK_CONTROL, OCF: OCF_LINK_CONTROL_LINK_KEY_REQUEST_NEG_REPLY},
|
|
|
|
ParameterLength: sizeof(BT_HCICommand_LinkKeyNAKResp_t),
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Copy over the temporary BT device address saved from the Link Key Request event */
|
|
|
|
BT_HCICommand_LinkKeyNAKResp_t LinkKeyNAKParams;
|
|
|
|
memcpy(LinkKeyNAKParams.RemoteAddress, Bluetooth_TempDeviceAddress, sizeof(LinkKeyNAKParams.RemoteAddress));
|
|
|
|
|
|
|
|
/* Send the command to transmit the link key NAK to the receiver */
|
|
|
|
Bluetooth_SendHCICommand(&HCICommandHeader, &LinkKeyNAKParams, sizeof(BT_HCICommand_LinkKeyNAKResp_t));
|
|
|
|
|
|
|
|
Bluetooth_HCIProcessingState = Bluetooth_ProcessEvents;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Sends a Bluetooth HCI control command to the attached Bluetooth device.
|
|
|
|
*
|
|
|
|
* \param[in] HCICommandHeader HCI command header to send to the attached device
|
|
|
|
* \param[in] Parameters Pointer to the source of the control parameters (if any)
|
|
|
|
* \param[in] ParameterLength Length of the parameters to send in bytes
|
|
|
|
*
|
|
|
|
* \return A value from the USB_Host_SendControlErrorCodes_t enum.
|
|
|
|
*/
|
|
|
|
static uint8_t Bluetooth_SendHCICommand(const BT_HCICommand_Header_t* const HCICommandHeader, const void* Parameters, const uint16_t ParameterLength)
|
|
|
|
{
|
|
|
|
/* Need to reserve the amount of bytes given in the header for the complete payload */
|
|
|
|
uint8_t CommandBuffer[sizeof(BT_HCICommand_Header_t) + HCICommandHeader->ParameterLength];
|
|
|
|
|
|
|
|
USB_ControlRequest = (USB_Request_Header_t)
|
|
|
|
{
|
|
|
|
.bmRequestType = (REQDIR_HOSTTODEVICE | REQTYPE_CLASS | REQREC_DEVICE),
|
|
|
|
.bRequest = 0,
|
|
|
|
.wValue = 0,
|
|
|
|
.wIndex = 0,
|
|
|
|
.wLength = sizeof(CommandBuffer)
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Copy over the HCI command header to the allocated buffer */
|
|
|
|
memcpy(CommandBuffer, HCICommandHeader, sizeof(BT_HCICommand_Header_t));
|
|
|
|
|
|
|
|
/* Zero out the parameter section of the response so that all padding bytes are known to be zero */
|
|
|
|
memset(&CommandBuffer[sizeof(BT_HCICommand_Header_t)], 0x00, HCICommandHeader->ParameterLength);
|
|
|
|
|
|
|
|
/* Copy over the command parameters (if any) to the command buffer - note, the number of actual source parameter bytes
|
|
|
|
may differ to those in the header; any difference in length is filled with 0x00 padding bytes */
|
|
|
|
memcpy(&CommandBuffer[sizeof(BT_HCICommand_Header_t)], Parameters, ParameterLength);
|
|
|
|
|
|
|
|
Pipe_SelectPipe(PIPE_CONTROLPIPE);
|
|
|
|
return USB_Host_SendControlRequest(CommandBuffer);
|
|
|
|
}
|