From 2e36e02a2616d5e5cce87836b31c7cd43786843a Mon Sep 17 00:00:00 2001 From: "matthias.ringwald" Date: Fri, 8 May 2009 21:37:54 +0000 Subject: [PATCH] minimal h4 statemachine, reset + inq demo --- project.xcodeproj/project.pbxproj | 4 +- src/hci_transport.h | 10 +- src/hci_transport_h4.c | 177 +++++++++++++++++++++++++----- src/main.c | 53 ++++++++- 4 files changed, 207 insertions(+), 37 deletions(-) diff --git a/project.xcodeproj/project.pbxproj b/project.xcodeproj/project.pbxproj index e1e8c9d59..60eeda577 100644 --- a/project.xcodeproj/project.pbxproj +++ b/project.xcodeproj/project.pbxproj @@ -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 */, ); diff --git a/src/hci_transport.h b/src/hci_transport.h index 9eb03726a..e9e3fdcf6 100644 --- a/src/hci_transport.h +++ b/src/hci_transport.h @@ -8,13 +8,15 @@ */ #pragma once +#include + 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)(); diff --git a/src/hci_transport_h4.c b/src/hci_transport_h4.c index 5b3ec4613..8502d7766 100644 --- a/src/hci_transport_h4.c +++ b/src/hci_transport_h4.c @@ -4,61 +4,186 @@ * Created by Matthias Ringwald on 4/29/09. * */ +#include /* POSIX terminal control definitions */ +#include /* File control definitions */ +#include /* UNIX standard function definitions */ +#include #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 }; diff --git a/src/main.c b/src/main.c index c45d820fb..7ec334da5 100644 --- a/src/main.c +++ b/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; isend_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; }