mirror of
https://github.com/bluekitchen/btstack.git
synced 2025-03-27 23:37:25 +00:00
minimal h4 statemachine, reset + inq demo
This commit is contained in:
parent
475c8125ef
commit
2e36e02a26
@ -58,10 +58,10 @@
|
||||
08FB7795FE84155DC02AAC07 /* Source */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
9C46FC340FA906F700ABEF05 /* hci.c */,
|
||||
9C46FC350FA906F700ABEF05 /* hci.h */,
|
||||
9C46FC360FA906F700ABEF05 /* hci_transport_h4.c */,
|
||||
9C46FC340FA906F700ABEF05 /* hci.c */,
|
||||
9C46FC370FA906F700ABEF05 /* hci_transport_h4.h */,
|
||||
9C46FC360FA906F700ABEF05 /* hci_transport_h4.c */,
|
||||
9C46FC380FA906F700ABEF05 /* hci_transport.h */,
|
||||
9C46FBAE0FA8F31800ABEF05 /* main.c */,
|
||||
);
|
||||
|
@ -8,13 +8,15 @@
|
||||
*/
|
||||
#pragma once
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
typedef struct {
|
||||
int (*open)(void *transport_config);
|
||||
int (*close)();
|
||||
int (*send_cmd_packet)(void *packet, int size);
|
||||
int (*send_acl_packet)(void *packet, int size);
|
||||
void (*register_event_packet_handler)(void (*handler)(void *packet, int size));
|
||||
void (*register_acl_packet_handler) (void (*handler)(void *packet, int size));
|
||||
int (*send_cmd_packet)(uint8_t *packet, int size);
|
||||
int (*send_acl_packet)(uint8_t *packet, int size);
|
||||
void (*register_event_packet_handler)(void (*handler)(uint8_t *packet, int size));
|
||||
void (*register_acl_packet_handler) (void (*handler)(uint8_t *packet, int size));
|
||||
int (*get_fd)(); // <-- only used for select(..) call
|
||||
int (*handle_data)(); // -- to be called when select( .. ) returns for the fd
|
||||
const char * (*get_transport_name)();
|
||||
|
@ -4,61 +4,186 @@
|
||||
* Created by Matthias Ringwald on 4/29/09.
|
||||
*
|
||||
*/
|
||||
#include <termios.h> /* POSIX terminal control definitions */
|
||||
#include <fcntl.h> /* File control definitions */
|
||||
#include <unistd.h> /* UNIX standard function definitions */
|
||||
#include <stdio.h>
|
||||
|
||||
#include "hci_transport_h4.h"
|
||||
|
||||
typedef enum {
|
||||
H4_W4_PACKET_TYPE,
|
||||
H4_W4_EVENT_HEADER,
|
||||
H4_W4_EVENT_PAYLOAD,
|
||||
H4_W4_ACL_HEADER,
|
||||
H4_W4_ACL_PAYLOAD
|
||||
} H4_STATE;
|
||||
|
||||
static void dummy_handler(uint8_t *packet, int size);
|
||||
|
||||
// typedef struct {
|
||||
// static hci_uart_config_t *config;
|
||||
static int fd;
|
||||
static void (*event_packet_handler)(uint8_t *packet, int size) = dummy_handler;
|
||||
static void (*acl_packet_handler)(uint8_t *packet, int size) = dummy_handler;
|
||||
static H4_STATE h4_state;
|
||||
static int bytes_to_read;
|
||||
static int read_pos;
|
||||
static uint8_t hci_event_buffer[255+2]; // maximal payload + 2 bytes header
|
||||
// } hci_h4_transport_private;
|
||||
|
||||
|
||||
|
||||
// prototypes
|
||||
static int open(void *transport_config){
|
||||
static int h4_open(void *transport_config){
|
||||
hci_uart_config_t * uart_config = (hci_uart_config_t*) transport_config;
|
||||
struct termios toptions;
|
||||
fd = open(uart_config->device_name, O_RDWR | O_NOCTTY | O_NDELAY);
|
||||
if (fd == -1) {
|
||||
perror("init_serialport: Unable to open port ");
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (tcgetattr(fd, &toptions) < 0) {
|
||||
perror("init_serialport: Couldn't get term attributes");
|
||||
return -1;
|
||||
}
|
||||
speed_t brate = uart_config->baudrate; // let you override switch below if needed
|
||||
switch(uart_config->baudrate) {
|
||||
case 57600: brate=B57600; break;
|
||||
case 115200: brate=B115200; break;
|
||||
}
|
||||
cfsetispeed(&toptions, brate);
|
||||
cfsetospeed(&toptions, brate);
|
||||
|
||||
// 8N1
|
||||
toptions.c_cflag &= ~PARENB;
|
||||
toptions.c_cflag &= ~CSTOPB;
|
||||
toptions.c_cflag &= ~CSIZE;
|
||||
toptions.c_cflag |= CS8;
|
||||
|
||||
if (uart_config->flowcontrol) {
|
||||
// with flow control
|
||||
toptions.c_cflag |= CRTSCTS;
|
||||
} else {
|
||||
// no flow control
|
||||
toptions.c_cflag &= ~CRTSCTS;
|
||||
}
|
||||
|
||||
toptions.c_cflag |= CREAD | CLOCAL; // turn on READ & ignore ctrl lines
|
||||
toptions.c_iflag &= ~(IXON | IXOFF | IXANY); // turn off s/w flow ctrl
|
||||
|
||||
toptions.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG); // make raw
|
||||
toptions.c_oflag &= ~OPOST; // make raw
|
||||
|
||||
// see: http://unixwiz.net/techtips/termios-vmin-vtime.html
|
||||
toptions.c_cc[VMIN] = 1;
|
||||
toptions.c_cc[VTIME] = 0;
|
||||
|
||||
if( tcsetattr(fd, TCSANOW, &toptions) < 0) {
|
||||
perror("init_serialport: Couldn't set term attributes");
|
||||
return -1;
|
||||
}
|
||||
|
||||
// init state machine
|
||||
bytes_to_read = 1;
|
||||
h4_state = H4_W4_PACKET_TYPE;
|
||||
read_pos = 0;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int close(){
|
||||
static int h4_close(){
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int send_cmd_packet(void *packet, int size){
|
||||
static int h4_send_cmd_packet(uint8_t *packet, int size){
|
||||
char *data = (char*) packet;
|
||||
char cmd_type = 1;
|
||||
write(fd, &cmd_type, 1);
|
||||
while (size > 0) {
|
||||
int bytes_written = write(fd, data, size);
|
||||
if (bytes_written < 0) {
|
||||
return bytes_written;
|
||||
}
|
||||
data += bytes_written;
|
||||
size -= bytes_written;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int send_acl_packet(void *packet, int size){
|
||||
static int h4_send_acl_packet(uint8_t *packet, int size){
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void register_event_packet_handler(void (*handler)(void *packet, int size)){
|
||||
static void h4_register_event_packet_handler(void (*handler)(uint8_t *packet, int size)){
|
||||
event_packet_handler = handler;
|
||||
}
|
||||
|
||||
static void register_acl_packet_handler (void (*handler)(void *packet, int size)){
|
||||
static void h4_register_acl_packet_handler (void (*handler)(uint8_t *packet, int size)){
|
||||
acl_packet_handler = handler;
|
||||
}
|
||||
|
||||
static int get_fd() {
|
||||
return -1;
|
||||
static int h4_get_fd() {
|
||||
return fd;
|
||||
}
|
||||
|
||||
static int handle_data() {
|
||||
static int h4_handle_data() {
|
||||
// read up to bytes_to_read data in
|
||||
int bytes_read = read(fd, &hci_event_buffer[read_pos], bytes_to_read);
|
||||
if (bytes_read < 0) {
|
||||
return bytes_read;
|
||||
}
|
||||
// printf("Bytes read: %u\n", bytes_read);
|
||||
bytes_to_read -= bytes_read;
|
||||
read_pos += bytes_read;
|
||||
if (bytes_to_read > 0) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
// act
|
||||
switch (h4_state) {
|
||||
case H4_W4_PACKET_TYPE:
|
||||
if (hci_event_buffer[0] == 4){
|
||||
read_pos = 0;
|
||||
bytes_to_read = 2;
|
||||
h4_state = H4_W4_EVENT_HEADER;
|
||||
}
|
||||
break;
|
||||
case H4_W4_EVENT_HEADER:
|
||||
bytes_to_read = hci_event_buffer[1];
|
||||
h4_state = H4_W4_EVENT_PAYLOAD;
|
||||
break;
|
||||
case H4_W4_EVENT_PAYLOAD:
|
||||
event_packet_handler(hci_event_buffer, read_pos);
|
||||
h4_state = H4_W4_PACKET_TYPE;
|
||||
read_pos = 0;
|
||||
bytes_to_read = 1;
|
||||
break;
|
||||
case H4_W4_ACL_HEADER:
|
||||
break;
|
||||
case H4_W4_ACL_PAYLOAD:
|
||||
break;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const char * get_transport_name(){
|
||||
static const char * h4_get_transport_name(){
|
||||
return "H4";
|
||||
}
|
||||
|
||||
// private data
|
||||
|
||||
typedef struct {
|
||||
hci_uart_config_t *config;
|
||||
void (*event_packet_handle)(void *packet, int size);
|
||||
void (*acl_packet_handle)(void *packet, int size);
|
||||
} hci_h4_transport_private_t;
|
||||
static void dummy_handler(uint8_t *packet, int size){
|
||||
}
|
||||
|
||||
// single instance
|
||||
hci_transport_t hci_h4_transport = {
|
||||
open,
|
||||
close,
|
||||
send_cmd_packet,
|
||||
send_acl_packet,
|
||||
register_acl_packet_handler,
|
||||
register_event_packet_handler,
|
||||
get_fd,
|
||||
handle_data,
|
||||
get_transport_name
|
||||
h4_open,
|
||||
h4_close,
|
||||
h4_send_cmd_packet,
|
||||
h4_send_acl_packet,
|
||||
h4_register_event_packet_handler,
|
||||
h4_register_acl_packet_handler,
|
||||
h4_get_fd,
|
||||
h4_handle_data,
|
||||
h4_get_transport_name
|
||||
};
|
||||
|
53
src/main.c
53
src/main.c
@ -18,7 +18,6 @@
|
||||
static hci_transport_t * transport;
|
||||
static hci_uart_config_t config;
|
||||
|
||||
|
||||
#if 0
|
||||
static void *hci_daemon_thread(void *arg){
|
||||
printf("HCI Daemon started\n");
|
||||
@ -27,6 +26,26 @@ static void *hci_daemon_thread(void *arg){
|
||||
}
|
||||
#endif
|
||||
|
||||
void hexdump(uint8_t *data, int size){
|
||||
int i;
|
||||
for (i=0; i<size;i++){
|
||||
printf("%02X ", data[i]);
|
||||
}
|
||||
printf("\n");
|
||||
}
|
||||
|
||||
void event_handler(uint8_t *packet, int size){
|
||||
printf("Event received, size %u\n", size );
|
||||
hexdump( packet, size);
|
||||
|
||||
//
|
||||
if (packet[3] == 3 && packet[4] == 12){
|
||||
// reset done, send inq
|
||||
uint8_t inq[] = { 0x01, 0x04, 0x05, 0x33, 0x8b, 0x9e, 0x0a, 0x14};
|
||||
transport->send_cmd_packet( inq, sizeof(inq) );
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
int main (int argc, const char * argv[]) {
|
||||
|
||||
@ -57,11 +76,35 @@ int main (int argc, const char * argv[]) {
|
||||
|
||||
#endif
|
||||
|
||||
hci_init(transport, &config);
|
||||
hci_power_control(HCI_POWER_ON);
|
||||
// hci_init(transport, &config);
|
||||
// hci_power_control(HCI_POWER_ON);
|
||||
|
||||
// open low-level device
|
||||
transport->open(&config);
|
||||
|
||||
//
|
||||
// register callback
|
||||
// register callbacks
|
||||
//
|
||||
hci_run();
|
||||
// hci_run();
|
||||
transport->register_event_packet_handler(&event_handler);
|
||||
|
||||
// get fd for select call
|
||||
int transport_fd = transport->get_fd();
|
||||
|
||||
// send hci reset
|
||||
uint8_t reset[] = { 0x03, 0x0c, 0x00 };
|
||||
transport->send_cmd_packet( reset, sizeof(reset) );
|
||||
|
||||
//
|
||||
fd_set descriptors;
|
||||
FD_ZERO(&descriptors);
|
||||
while (1){
|
||||
FD_SET(transport_fd, &descriptors);
|
||||
// int ready =
|
||||
select( transport_fd+1, &descriptors, NULL, NULL, NULL);
|
||||
// printf("Ready: %d, isset() = %u\n", ready, FD_ISSET(transport_fd, &descriptors));
|
||||
transport->handle_data();
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user