12531c97eSMatthias Ringwald /* 22531c97eSMatthias Ringwald * Copyright (C) 2014 BlueKitchen GmbH 32531c97eSMatthias Ringwald * 42531c97eSMatthias Ringwald * Redistribution and use in source and binary forms, with or without 52531c97eSMatthias Ringwald * modification, are permitted provided that the following conditions 62531c97eSMatthias Ringwald * are met: 72531c97eSMatthias Ringwald * 82531c97eSMatthias Ringwald * 1. Redistributions of source code must retain the above copyright 92531c97eSMatthias Ringwald * notice, this list of conditions and the following disclaimer. 102531c97eSMatthias Ringwald * 2. Redistributions in binary form must reproduce the above copyright 112531c97eSMatthias Ringwald * notice, this list of conditions and the following disclaimer in the 122531c97eSMatthias Ringwald * documentation and/or other materials provided with the distribution. 132531c97eSMatthias Ringwald * 3. Neither the name of the copyright holders nor the names of 142531c97eSMatthias Ringwald * contributors may be used to endorse or promote products derived 152531c97eSMatthias Ringwald * from this software without specific prior written permission. 162531c97eSMatthias Ringwald * 4. Any redistribution, use, or modification is done solely for 172531c97eSMatthias Ringwald * personal benefit and not for any commercial purpose or for 182531c97eSMatthias Ringwald * monetary gain. 192531c97eSMatthias Ringwald * 202531c97eSMatthias Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 212531c97eSMatthias Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 222531c97eSMatthias Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 232531c97eSMatthias Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS 242531c97eSMatthias Ringwald * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 252531c97eSMatthias Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 262531c97eSMatthias Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 272531c97eSMatthias Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 282531c97eSMatthias Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 292531c97eSMatthias Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 302531c97eSMatthias Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 312531c97eSMatthias Ringwald * SUCH DAMAGE. 322531c97eSMatthias Ringwald * 332531c97eSMatthias Ringwald * Please inquire about commercial licensing options at 342531c97eSMatthias Ringwald * [email protected] 352531c97eSMatthias Ringwald * 362531c97eSMatthias Ringwald */ 372531c97eSMatthias Ringwald 382531c97eSMatthias Ringwald /* 392531c97eSMatthias Ringwald * rfcomm_echo.c 402531c97eSMatthias Ringwald */ 412531c97eSMatthias Ringwald 422531c97eSMatthias Ringwald #include <unistd.h> 432531c97eSMatthias Ringwald #include <stdio.h> 442531c97eSMatthias Ringwald #include <stdlib.h> 452531c97eSMatthias Ringwald #include <strings.h> 462531c97eSMatthias Ringwald #include <errno.h> 472531c97eSMatthias Ringwald #include <string.h> 482531c97eSMatthias Ringwald #include <fcntl.h> 492531c97eSMatthias Ringwald #include <sys/types.h> 502531c97eSMatthias Ringwald #include <sys/stat.h> 512531c97eSMatthias Ringwald 522531c97eSMatthias Ringwald #include "btstack_client.h" 532531c97eSMatthias Ringwald #include "btstack_run_loop_posix.h" 542531c97eSMatthias Ringwald #include "classic/sdp_util.h" 552531c97eSMatthias Ringwald 562531c97eSMatthias Ringwald #define NUM_ROWS 25 572531c97eSMatthias Ringwald #define NUM_COLS 80 582531c97eSMatthias Ringwald 592531c97eSMatthias Ringwald // input from command line arguments 602531c97eSMatthias Ringwald bd_addr_t addr = { }; 61711e6c80SMatthias Ringwald hci_con_handle_t con_handle; 622531c97eSMatthias Ringwald char pin[17]; 632531c97eSMatthias Ringwald int counter = 0; 642531c97eSMatthias Ringwald uint16_t rfcomm_channel_id = 0; 652531c97eSMatthias Ringwald uint16_t mtu = 0; 662531c97eSMatthias Ringwald static uint8_t spp_service_buffer[150]; 672531c97eSMatthias Ringwald uint8_t test_data[NUM_ROWS * NUM_COLS + 1]; 682531c97eSMatthias Ringwald 692531c97eSMatthias Ringwald void create_test_data(void){ 702531c97eSMatthias Ringwald int x,y; 712531c97eSMatthias Ringwald for (y=0;y<NUM_ROWS;y++){ 722531c97eSMatthias Ringwald for (x=0;x<NUM_COLS-2;x++){ 732531c97eSMatthias Ringwald test_data[y*NUM_COLS+x] = '0' + (x % 10); 742531c97eSMatthias Ringwald } 752531c97eSMatthias Ringwald test_data[y*NUM_COLS+NUM_COLS-2] = '\n'; 762531c97eSMatthias Ringwald test_data[y*NUM_COLS+NUM_COLS-1] = '\r'; 772531c97eSMatthias Ringwald } 782531c97eSMatthias Ringwald test_data[NUM_COLS*NUM_ROWS] = 0; 792531c97eSMatthias Ringwald } 802531c97eSMatthias Ringwald 812531c97eSMatthias Ringwald void packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 822531c97eSMatthias Ringwald bd_addr_t event_addr; 832531c97eSMatthias Ringwald uint16_t rfcomm_channel_nr; 842531c97eSMatthias Ringwald 852531c97eSMatthias Ringwald switch (packet_type) { 862531c97eSMatthias Ringwald 872531c97eSMatthias Ringwald case RFCOMM_DATA_PACKET: 882531c97eSMatthias Ringwald printf("Received RFCOMM data on channel id %u, size %u\n", channel, size); 892531c97eSMatthias Ringwald printf_hexdump(packet, size); 902531c97eSMatthias Ringwald bt_send_rfcomm(channel, packet, size); 912531c97eSMatthias Ringwald break; 922531c97eSMatthias Ringwald 932531c97eSMatthias Ringwald case HCI_EVENT_PACKET: 940e2df43fSMatthias Ringwald switch (hci_event_packet_get_type(packet)) { 952531c97eSMatthias Ringwald 962531c97eSMatthias Ringwald case BTSTACK_EVENT_POWERON_FAILED: 972531c97eSMatthias Ringwald // handle HCI init failure 982531c97eSMatthias Ringwald printf("HCI Init failed - make sure you have turned off Bluetooth in the System Settings\n"); 992531c97eSMatthias Ringwald exit(1); 1002531c97eSMatthias Ringwald break; 1012531c97eSMatthias Ringwald 1022531c97eSMatthias Ringwald case BTSTACK_EVENT_STATE: 1032531c97eSMatthias Ringwald // bt stack activated, get started 1042531c97eSMatthias Ringwald if (packet[2] == HCI_STATE_WORKING) { 1052531c97eSMatthias Ringwald // get persistent RFCOMM channel 1062531c97eSMatthias Ringwald printf("HCI_STATE_WORKING\n"); 1072531c97eSMatthias Ringwald bt_send_cmd(&rfcomm_persistent_channel_for_service_cmd, "ch.ringwald.btstack.rfcomm_test"); 1082531c97eSMatthias Ringwald } 1092531c97eSMatthias Ringwald break; 1102531c97eSMatthias Ringwald 1112531c97eSMatthias Ringwald case RFCOMM_EVENT_PERSISTENT_CHANNEL: 1122531c97eSMatthias Ringwald rfcomm_channel_nr = packet[3]; 1132531c97eSMatthias Ringwald printf("RFCOMM channel %u was assigned by BTdaemon\n", rfcomm_channel_nr); 1142531c97eSMatthias Ringwald bt_send_cmd(&rfcomm_register_service_cmd, rfcomm_channel_nr, 0xffff); // reserved channel, mtu limited by l2cap 1152531c97eSMatthias Ringwald break; 1162531c97eSMatthias Ringwald 1172531c97eSMatthias Ringwald case RFCOMM_EVENT_SERVICE_REGISTERED: 1182531c97eSMatthias Ringwald printf("RFCOMM_EVENT_SERVICE_REGISTERED\n"); 1192531c97eSMatthias Ringwald rfcomm_channel_nr = packet[3]; 1202531c97eSMatthias Ringwald // register SDP for our SPP 121*efda0b48SMatthias Ringwald spp_create_sdp_record((uint8_t*) spp_service_buffer, 0x10001, rfcomm_channel_nr, "RFCOMM Test"); 1222531c97eSMatthias Ringwald bt_send_cmd(&sdp_register_service_record_cmd, spp_service_buffer); 1232531c97eSMatthias Ringwald bt_send_cmd(&btstack_set_discoverable, 1); 1242531c97eSMatthias Ringwald break; 1252531c97eSMatthias Ringwald 1262531c97eSMatthias Ringwald case RFCOMM_EVENT_CREDITS: 1272531c97eSMatthias Ringwald sprintf((char*)test_data, "\n\r\n\r-> %09u <- ", counter++); 1282531c97eSMatthias Ringwald bt_send_rfcomm(rfcomm_channel_id, test_data, mtu); 1292531c97eSMatthias Ringwald break; 1302531c97eSMatthias Ringwald 1312531c97eSMatthias Ringwald case HCI_EVENT_PIN_CODE_REQUEST: 1322531c97eSMatthias Ringwald // inform about pin code request 1332531c97eSMatthias Ringwald printf("Using PIN 0000\n"); 1342531c97eSMatthias Ringwald reverse_bd_addr(&packet[2], 1352531c97eSMatthias Ringwald event_addr); 1362531c97eSMatthias Ringwald bt_send_cmd(&hci_pin_code_request_reply, &event_addr, 4, "0000"); 1372531c97eSMatthias Ringwald break; 1382531c97eSMatthias Ringwald 1392531c97eSMatthias Ringwald case RFCOMM_EVENT_INCOMING_CONNECTION: 1402531c97eSMatthias Ringwald // data: event (8), len(8), address(48), channel (8), rfcomm_cid (16) 1412531c97eSMatthias Ringwald reverse_bd_addr(&packet[2], 1422531c97eSMatthias Ringwald event_addr); 1432531c97eSMatthias Ringwald rfcomm_channel_nr = packet[8]; 1442531c97eSMatthias Ringwald rfcomm_channel_id = little_endian_read_16(packet, 9); 1452531c97eSMatthias Ringwald printf("RFCOMM channel %u requested for %s\n", rfcomm_channel_nr, bd_addr_to_str(event_addr)); 1462531c97eSMatthias Ringwald bt_send_cmd(&rfcomm_accept_connection_cmd, rfcomm_channel_id); 1472531c97eSMatthias Ringwald break; 1482531c97eSMatthias Ringwald 1492531c97eSMatthias Ringwald case RFCOMM_EVENT_OPEN_CHANNEL_COMPLETE: 1502531c97eSMatthias Ringwald // data: event(8), len(8), status (8), address (48), handle(16), server channel(8), rfcomm_cid(16), max frame size(16) 1512531c97eSMatthias Ringwald if (packet[2]) { 1522531c97eSMatthias Ringwald printf("RFCOMM channel open failed, status %u\n", packet[2]); 1532531c97eSMatthias Ringwald } else { 1542531c97eSMatthias Ringwald // data: event(8), len(8), status (8), address (48), handle (16), server channel(8), rfcomm_cid(16), max frame size(16) 1552531c97eSMatthias Ringwald rfcomm_channel_id = little_endian_read_16(packet, 12); 1562531c97eSMatthias Ringwald mtu = little_endian_read_16(packet, 14); 1572531c97eSMatthias Ringwald 1582531c97eSMatthias Ringwald printf("RFCOMM channel open succeeded. New RFCOMM Channel ID %u, max frame size %u\n", rfcomm_channel_id, mtu); 1592531c97eSMatthias Ringwald } 1602531c97eSMatthias Ringwald break; 1612531c97eSMatthias Ringwald 1622531c97eSMatthias Ringwald case HCI_EVENT_DISCONNECTION_COMPLETE: 1632531c97eSMatthias Ringwald // connection closed -> quit test app 1642531c97eSMatthias Ringwald printf("Basebank connection closed\n"); 1652531c97eSMatthias Ringwald break; 1662531c97eSMatthias Ringwald 1672531c97eSMatthias Ringwald default: 1682531c97eSMatthias Ringwald break; 1692531c97eSMatthias Ringwald } 1702531c97eSMatthias Ringwald break; 1712531c97eSMatthias Ringwald default: 1722531c97eSMatthias Ringwald break; 1732531c97eSMatthias Ringwald } 1742531c97eSMatthias Ringwald } 1752531c97eSMatthias Ringwald 1762531c97eSMatthias Ringwald 1772531c97eSMatthias Ringwald int main (int argc, const char * argv[]){ 1782531c97eSMatthias Ringwald 1792531c97eSMatthias Ringwald create_test_data(); 1802531c97eSMatthias Ringwald printf("created test data: \n%s\n", test_data); 1812531c97eSMatthias Ringwald 1822531c97eSMatthias Ringwald btstack_run_loop_init(btstack_run_loop_posix_get_instance()); 1832531c97eSMatthias Ringwald int err = bt_open(); 1842531c97eSMatthias Ringwald if (err) { 1852531c97eSMatthias Ringwald fprintf(stderr,"Failed to open connection to BTdaemon, err %d\n",err); 1862531c97eSMatthias Ringwald return 1; 1872531c97eSMatthias Ringwald } 1882531c97eSMatthias Ringwald bt_register_packet_handler(packet_handler); 1892531c97eSMatthias Ringwald 1902531c97eSMatthias Ringwald bt_send_cmd(&btstack_set_power_mode, HCI_POWER_ON ); 1912531c97eSMatthias Ringwald btstack_run_loop_execute(); 1922531c97eSMatthias Ringwald bt_close(); 1932531c97eSMatthias Ringwald return 0; 1942531c97eSMatthias Ringwald } 195