From 751081ab186d512093c82dc9a2d14a3533a11a56 Mon Sep 17 00:00:00 2001 From: Taejin Woo Date: Thu, 2 Jul 2015 17:31:13 +0900 Subject: [PATCH 01/16] Remove the function including un-used vconf key Change-Id: I87302916bcb2d44d86c2c07d371d1045a05abc38 Signed-off-by: Taejin Woo --- set-address/setbd.c | 65 ++--------------------------------------------------- 1 file changed, 2 insertions(+), 63 deletions(-) diff --git a/set-address/setbd.c b/set-address/setbd.c index 092da16..f4497eb 100644 --- a/set-address/setbd.c +++ b/set-address/setbd.c @@ -196,65 +196,6 @@ int readBDaddrTI(void){ return 0; } #endif -int make_bt_address_from_tapi_imei(unsigned char * bt_address) -{ - char * temp=NULL; - int tapi_state=0; - int ret=-1; - int i=0; - - if(bt_address==NULL) - return -EBADR; - - ret=vconf_get_int(VCONFKEY_TELEPHONY_TAPI_STATE,&tapi_state); - if(tapi_state==VCONFKEY_TELEPHONY_TAPI_STATE_READY && ret==0){ - temp=vconf_get_str(VCONFKEY_TELEPHONY_IMEI); - APP_DEBUG("TAPI_IMEI: %s\n",temp); - -#ifdef IMEI_BASED_RAND_FEATURE - if(strcmp(temp,DEFAULT_IMEI)==0){ - APP_DEBUG("TAPI_IMEI is defulat IMEI\n"); - is_default_imei=TRUE; - return -ENODATA; - } -#else - APP_DEBUG("Temporarily we skip reading TAPI_IMEI\n"); - APP_DEBUG(" due to TAPI IMEI API is deprecated\n"); - is_default_imei=TRUE; - return -ENODATA; -#endif - - if(strcmp(temp,"")==0) - return -ENODATA; - - if(strlen(temp)<14) - return -ENODATA; - - memcpy(bt_address, BD_PREFIX, 5); - - for(i=5 ;i<14;i++){ - if(i==7){ - bt_address[i]='\n'; - continue; - } - - bt_address[i]=temp[i]; - } - - }else{ - APP_DEBUG("TAPI_IMEI Reading Error\n"); - return -ENODATA; - } - - APP_DEBUG("Bluetooth Address\n"); - for(i=0;i Date: Thu, 2 Jul 2015 17:48:52 +0900 Subject: [PATCH 02/16] Remove vconf value Change-Id: I9551d5f47eff418d780a444563f3e9b823e8e576 Signed-off-by: Taejin Woo --- set-address/setbd.c | 30 ------------------------------ 1 file changed, 30 deletions(-) diff --git a/set-address/setbd.c b/set-address/setbd.c index f4497eb..013a92f 100644 --- a/set-address/setbd.c +++ b/set-address/setbd.c @@ -293,32 +293,6 @@ int make_bt_address(gboolean overwrite_bt_address) #endif } -void vconf_cb(keynode_t *key, void * data) -{ - char * key_string=NULL; - - switch(vconf_keynode_get_type(key)) - { - case VCONF_TYPE_STRING: - key_string=vconf_keynode_get_str(key); - if(strcmp(key_string,"")!=0) - { - APP_DEBUG("Vconf Call back trial\n"); - /* This case means TAPI writes IMEI correctly */ - /* Because we write BT address which comes from IMEI again */ - make_bt_address(TRUE); - g_main_loop_quit(loop); - } - - break; - - - default: - break; - } - return; -} - gboolean exit_cb(gpointer data) { @@ -340,12 +314,8 @@ int main() exit(0); #ifdef IMEI_BASED_RAND_FEATURE - vconf_notify_key_changed(VCONFKEY_TELEPHONY_IMEI,vconf_cb,NULL); - g_timeout_add_seconds(10,exit_cb,NULL); g_main_loop_run(loop); - - vconf_ignore_key_changed(VCONFKEY_TELEPHONY_IMEI,vconf_cb); #endif return 0; -- 2.7.4 From 57c2f185918e1c92f2b8ca34f2171397bd5e8c66 Mon Sep 17 00:00:00 2001 From: DoHyun Pyun Date: Wed, 8 Jul 2015 17:28:09 +0900 Subject: [PATCH 03/16] Fix the file path for hciconfig / hciattach Change-Id: I016be8073fd7e12dfd9630d73177d20730c94f39 Signed-off-by: DoHyun Pyun --- scripts/bt-dev-end.sh | 2 +- scripts/bt-dev-start.sh | 12 ++++++------ 2 files changed, 7 insertions(+), 7 deletions(-) diff --git a/scripts/bt-dev-end.sh b/scripts/bt-dev-end.sh index ef4731e..5fe10a9 100755 --- a/scripts/bt-dev-end.sh +++ b/scripts/bt-dev-end.sh @@ -5,7 +5,7 @@ # # Device down -/usr/sbin/hciconfig hci0 down +/usr/bin/hciconfig hci0 down # OMAP4 REVISION_NUM=`grep Revision /proc/cpuinfo | awk "{print \\$3}"` diff --git a/scripts/bt-dev-start.sh b/scripts/bt-dev-start.sh index 42d4d3b..de9e4e4 100755 --- a/scripts/bt-dev-start.sh +++ b/scripts/bt-dev-start.sh @@ -34,9 +34,9 @@ fi rfkill unblock bluetooth echo "Check for Bluetooth device status" -if (/usr/sbin/hciconfig | grep hci); then +if (/usr/bin/hciconfig | grep hci); then echo "Bluetooth device is UP" - /usr/sbin/hciconfig hci0 up + /usr/bin/hciconfig hci0 up else echo "Bluetooth device is DOWN" echo "Registering Bluetooth device" @@ -44,11 +44,11 @@ else $BCM_TOOL $BT_UART_DEVICE -FILE=/usr/etc/bluetooth/$BCM_FIRMWARE -BAUD=$UART_SPEED -ADDR=/opt/etc/.bd_addr -SETSCO=0,0,0,0,0,0,0,3,3,0 -LP > /dev/null 2>&1 # Attaching Broadcom device - if (/usr/sbin/hciattach $BT_UART_DEVICE -s $UART_SPEED $BT_CHIP_TYPE $UART_SPEED flow); then + if (/usr/bin/hciattach $BT_UART_DEVICE -s $UART_SPEED $BT_CHIP_TYPE $UART_SPEED flow); then sleep 0.1 - /usr/sbin/hciconfig hci0 up - /usr/sbin/hciconfig hci0 name $BT_PLATFORM_DEFAULT_HCI_NAME - /usr/sbin/hciconfig hci0 sspmode 1 + /usr/bin/hciconfig hci0 up + /usr/bin/hciconfig hci0 name $BT_PLATFORM_DEFAULT_HCI_NAME + /usr/bin/hciconfig hci0 sspmode 1 echo "HCIATTACH success" else echo "HCIATTACH failed" -- 2.7.4 From 836afb4f19a5888434764320b45137b3884d07cd Mon Sep 17 00:00:00 2001 From: DoHyun Pyun Date: Thu, 9 Jul 2015 19:51:00 +0900 Subject: [PATCH 04/16] Fix BT enable problem using BT API Change-Id: Id80483481da9ea207c57cfef60dcdf579c9c6f17 Signed-off-by: DoHyun Pyun --- scripts/bt-dev-end.sh | 2 +- scripts/bt-dev-start.sh | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/scripts/bt-dev-end.sh b/scripts/bt-dev-end.sh index 5fe10a9..0fbf2f8 100755 --- a/scripts/bt-dev-end.sh +++ b/scripts/bt-dev-end.sh @@ -20,7 +20,7 @@ fi killall hciattach # Turn off Bluetooth Chip -rfkill block bluetooth +/usr/sbin/rfkill block bluetooth #if [ -e /sys/class/gpio/gpio17/value ] #then diff --git a/scripts/bt-dev-start.sh b/scripts/bt-dev-start.sh index de9e4e4..2519604 100755 --- a/scripts/bt-dev-start.sh +++ b/scripts/bt-dev-start.sh @@ -31,7 +31,7 @@ then fi # Trun-on Bluetooth Chip -rfkill unblock bluetooth +/usr/sbin/rfkill unblock bluetooth echo "Check for Bluetooth device status" if (/usr/bin/hciconfig | grep hci); then @@ -52,6 +52,6 @@ else echo "HCIATTACH success" else echo "HCIATTACH failed" - rfkill block bluetooth + /usr/sbin/rfkill block bluetooth fi fi -- 2.7.4 From e29e186d496d43eb6fa3d625f2d9a1331532d795 Mon Sep 17 00:00:00 2001 From: Sudha Bheemanna Date: Thu, 17 Mar 2016 15:13:21 +0530 Subject: [PATCH 05/16] [Warnings] Fix build warnings. Fixed build time warnings in the code Change-Id: Ifab02bdbd82abe0f2feac374741a877b0c9946da Signed-off-by: Sudha Bheemanna --- CMakeLists.txt | 2 +- set-address/setbd.c | 8 +++++--- tools/bcmtool_4330b1.c | 27 ++++++++++++++++++--------- tools/bcmtool_4358a1.c | 27 ++++++++++++++++++--------- 4 files changed, 42 insertions(+), 22 deletions(-) mode change 100644 => 100755 set-address/setbd.c mode change 100644 => 100755 tools/bcmtool_4330b1.c mode change 100644 => 100755 tools/bcmtool_4358a1.c diff --git a/CMakeLists.txt b/CMakeLists.txt index c0285fa..919624a 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -7,7 +7,7 @@ FOREACH(flag ${package_CFLAGS}) SET(EXTRA_CFLAGS "${EXTRA_CFLAGS} ${flag}") ENDFOREACH(flag) -SET(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${EXTRA_CFLAGS}") +SET(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${EXTRA_CFLAGS} -fPIC -Wall -Werror") ADD_SUBDIRECTORY(set-address) ADD_SUBDIRECTORY(tools) diff --git a/set-address/setbd.c b/set-address/setbd.c old mode 100644 new mode 100755 index 013a92f..0fb6b83 --- a/set-address/setbd.c +++ b/set-address/setbd.c @@ -69,10 +69,9 @@ static gboolean is_default_imei=FALSE; #if defined(BT_CHIP_CSR) || defined(BT_CHIP_BROADCOM) int addremoveBD(char* path, char* pskey){ FILE *fd, *new; - int ret; char cmp[READ_BD_FILE_MAX]; char *result; - + int ret; fd = fopen(path, "r"); if(NULL == fd){ APP_DBG("Error open psr file\r\n"); @@ -87,6 +86,8 @@ int addremoveBD(char* path, char* pskey){ } ret = fputs(pskey,new); + if (ret < 0) + return -1; while(1){ result = fgets(cmp, READ_BD_FILE_MAX, fd); @@ -202,10 +203,11 @@ int make_bt_address(gboolean overwrite_bt_address) #if defined(BT_CHIP_CSR) || defined(BT_CHIP_BROADCOM) int fd; - int i; unsigned char txt[BD_ADDR_LEN]; unsigned char nap[4+1], uap[2+1], lap[6+1]; +#if defined(BT_CHIP_CSR) char pskey[PSKEY_LEN+3]; +#endif int ret; fd=open(BD_ADDR_FILE, O_RDONLY | O_SYNC); diff --git a/tools/bcmtool_4330b1.c b/tools/bcmtool_4330b1.c old mode 100644 new mode 100755 index 6350a06..e615376 --- a/tools/bcmtool_4330b1.c +++ b/tools/bcmtool_4330b1.c @@ -223,6 +223,7 @@ UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 *p_param_buf) { UINT8 pbuf[255] = {0,}; UINT8 i=0; + INT8 ret = 0; pbuf[0] = 0x1; pbuf[1] = (UINT8)(opcode); @@ -238,7 +239,10 @@ UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 *p_param_buf) dump(pbuf, param_len+4); - write(fd, pbuf, param_len+4); + ret = write(fd, pbuf, param_len+4); + if (ret != 0) + return -1; + return 0; } @@ -349,6 +353,7 @@ UINT8 DownloadPatchram( char *patchram1 ) INT32 FileSize=0; INT32 SentSize=0; + INT8 ret = 0; DEBUG1( "\n%s\n", patchram1); @@ -385,9 +390,13 @@ UINT8 DownloadPatchram( char *patchram1 ) len = buffer[3]; - fread(&buffer[4],sizeof(UINT8),len, pFile); + ret = fread(&buffer[4],sizeof(UINT8),len, pFile); + if (ret != 0) + return -1; - write(fd, buffer, len + 4); + ret = write(fd, buffer, len + 4); + if (ret != 0) + return -1; /* dispaly progress*/ SentSize += (len + 3); @@ -781,14 +790,14 @@ int main(int argc, char *argv[]) { char text[BTUI_MAX_STRING_LENGTH_PER_LINE]; - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); - sscanf(text,"%02x%02x",&bdaddr[0],&bdaddr[1]); + if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) != NULL) + sscanf(text,"%02x%02x",&bdaddr[0],&bdaddr[1]); - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); - sscanf(text,"%02x",&bdaddr[2]); + if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) != NULL) + sscanf(text,"%02x",&bdaddr[2]); - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); - sscanf(text,"%02x%02x%02x",&bdaddr[3],&bdaddr[4],&bdaddr[5]); + if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) != NULL) + sscanf(text,"%02x%02x%02x",&bdaddr[3],&bdaddr[4],&bdaddr[5]); fprintf(stderr,"Writing B/D Address = %02X:%02X:%02X:%02X:%02X:%02X\n",bdaddr[0],bdaddr[1],bdaddr[2],bdaddr[3],bdaddr[4],bdaddr[5]); diff --git a/tools/bcmtool_4358a1.c b/tools/bcmtool_4358a1.c old mode 100644 new mode 100755 index b35da8e..17dc7b4 --- a/tools/bcmtool_4358a1.c +++ b/tools/bcmtool_4358a1.c @@ -223,6 +223,7 @@ UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 *p_param_buf) { UINT8 pbuf[255] = {0,}; UINT8 i=0; + UINT8 ret = 0; pbuf[0] = 0x1; pbuf[1] = (UINT8)(opcode); @@ -238,7 +239,10 @@ UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 *p_param_buf) dump(pbuf, param_len+4); - write(fd, pbuf, param_len+4); + ret = write(fd, pbuf, param_len+4); + if (ret != 0) + return -1; + return 0; } @@ -349,6 +353,7 @@ UINT8 DownloadPatchram( char *patchram1 ) INT32 FileSize=0; INT32 SentSize=0; + INT8 ret = 0; DEBUG1( "\n%s\n", patchram1); @@ -385,9 +390,13 @@ UINT8 DownloadPatchram( char *patchram1 ) len = buffer[3]; - fread(&buffer[4],sizeof(UINT8),len, pFile); + ret = fread(&buffer[4],sizeof(UINT8),len, pFile); + if (ret != 0) + return -1; - write(fd, buffer, len + 4); + ret = write(fd, buffer, len + 4); + if (ret != 0) + return -1; /* dispaly progress*/ SentSize += (len + 3); @@ -781,14 +790,14 @@ int main(int argc, char *argv[]) { char text[BTUI_MAX_STRING_LENGTH_PER_LINE]; - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); - sscanf(text,"%02x%02x",&bdaddr[0],&bdaddr[1]); + if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile))!= NULL) + sscanf(text,"%02x%02x",&bdaddr[0],&bdaddr[1]); - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); - sscanf(text,"%02x",&bdaddr[2]); + if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) != NULL) + sscanf(text,"%02x",&bdaddr[2]); - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); - sscanf(text,"%02x%02x%02x",&bdaddr[3],&bdaddr[4],&bdaddr[5]); + if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) != NULL) + sscanf(text,"%02x%02x%02x",&bdaddr[3],&bdaddr[4],&bdaddr[5]); fprintf(stderr,"Writing B/D Address = %02X:%02X:%02X:%02X:%02X:%02X\n",bdaddr[0],bdaddr[1],bdaddr[2],bdaddr[3],bdaddr[4],bdaddr[5]); -- 2.7.4 From f497635a4f67917055edad6460b1ffbb331fd116 Mon Sep 17 00:00:00 2001 From: DoHyun Pyun Date: Mon, 11 Apr 2016 13:42:59 +0900 Subject: [PATCH 06/16] Revert "[Warnings] Fix build warnings." This reverts commit e29e186d496d43eb6fa3d625f2d9a1331532d795. Change-Id: I4d47613da072cb4b9a43e29d329f16d86aa1a58b Signed-off-by: DoHyun Pyun --- CMakeLists.txt | 2 +- set-address/setbd.c | 8 +++----- tools/bcmtool_4330b1.c | 27 +++++++++------------------ tools/bcmtool_4358a1.c | 27 +++++++++------------------ 4 files changed, 22 insertions(+), 42 deletions(-) mode change 100755 => 100644 set-address/setbd.c mode change 100755 => 100644 tools/bcmtool_4330b1.c mode change 100755 => 100644 tools/bcmtool_4358a1.c diff --git a/CMakeLists.txt b/CMakeLists.txt index 919624a..c0285fa 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -7,7 +7,7 @@ FOREACH(flag ${package_CFLAGS}) SET(EXTRA_CFLAGS "${EXTRA_CFLAGS} ${flag}") ENDFOREACH(flag) -SET(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${EXTRA_CFLAGS} -fPIC -Wall -Werror") +SET(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${EXTRA_CFLAGS}") ADD_SUBDIRECTORY(set-address) ADD_SUBDIRECTORY(tools) diff --git a/set-address/setbd.c b/set-address/setbd.c old mode 100755 new mode 100644 index 0fb6b83..013a92f --- a/set-address/setbd.c +++ b/set-address/setbd.c @@ -69,9 +69,10 @@ static gboolean is_default_imei=FALSE; #if defined(BT_CHIP_CSR) || defined(BT_CHIP_BROADCOM) int addremoveBD(char* path, char* pskey){ FILE *fd, *new; + int ret; char cmp[READ_BD_FILE_MAX]; char *result; - int ret; + fd = fopen(path, "r"); if(NULL == fd){ APP_DBG("Error open psr file\r\n"); @@ -86,8 +87,6 @@ int addremoveBD(char* path, char* pskey){ } ret = fputs(pskey,new); - if (ret < 0) - return -1; while(1){ result = fgets(cmp, READ_BD_FILE_MAX, fd); @@ -203,11 +202,10 @@ int make_bt_address(gboolean overwrite_bt_address) #if defined(BT_CHIP_CSR) || defined(BT_CHIP_BROADCOM) int fd; + int i; unsigned char txt[BD_ADDR_LEN]; unsigned char nap[4+1], uap[2+1], lap[6+1]; -#if defined(BT_CHIP_CSR) char pskey[PSKEY_LEN+3]; -#endif int ret; fd=open(BD_ADDR_FILE, O_RDONLY | O_SYNC); diff --git a/tools/bcmtool_4330b1.c b/tools/bcmtool_4330b1.c old mode 100755 new mode 100644 index e615376..6350a06 --- a/tools/bcmtool_4330b1.c +++ b/tools/bcmtool_4330b1.c @@ -223,7 +223,6 @@ UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 *p_param_buf) { UINT8 pbuf[255] = {0,}; UINT8 i=0; - INT8 ret = 0; pbuf[0] = 0x1; pbuf[1] = (UINT8)(opcode); @@ -239,10 +238,7 @@ UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 *p_param_buf) dump(pbuf, param_len+4); - ret = write(fd, pbuf, param_len+4); - if (ret != 0) - return -1; - + write(fd, pbuf, param_len+4); return 0; } @@ -353,7 +349,6 @@ UINT8 DownloadPatchram( char *patchram1 ) INT32 FileSize=0; INT32 SentSize=0; - INT8 ret = 0; DEBUG1( "\n%s\n", patchram1); @@ -390,13 +385,9 @@ UINT8 DownloadPatchram( char *patchram1 ) len = buffer[3]; - ret = fread(&buffer[4],sizeof(UINT8),len, pFile); - if (ret != 0) - return -1; + fread(&buffer[4],sizeof(UINT8),len, pFile); - ret = write(fd, buffer, len + 4); - if (ret != 0) - return -1; + write(fd, buffer, len + 4); /* dispaly progress*/ SentSize += (len + 3); @@ -790,14 +781,14 @@ int main(int argc, char *argv[]) { char text[BTUI_MAX_STRING_LENGTH_PER_LINE]; - if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) != NULL) - sscanf(text,"%02x%02x",&bdaddr[0],&bdaddr[1]); + fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + sscanf(text,"%02x%02x",&bdaddr[0],&bdaddr[1]); - if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) != NULL) - sscanf(text,"%02x",&bdaddr[2]); + fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + sscanf(text,"%02x",&bdaddr[2]); - if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) != NULL) - sscanf(text,"%02x%02x%02x",&bdaddr[3],&bdaddr[4],&bdaddr[5]); + fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + sscanf(text,"%02x%02x%02x",&bdaddr[3],&bdaddr[4],&bdaddr[5]); fprintf(stderr,"Writing B/D Address = %02X:%02X:%02X:%02X:%02X:%02X\n",bdaddr[0],bdaddr[1],bdaddr[2],bdaddr[3],bdaddr[4],bdaddr[5]); diff --git a/tools/bcmtool_4358a1.c b/tools/bcmtool_4358a1.c old mode 100755 new mode 100644 index 17dc7b4..b35da8e --- a/tools/bcmtool_4358a1.c +++ b/tools/bcmtool_4358a1.c @@ -223,7 +223,6 @@ UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 *p_param_buf) { UINT8 pbuf[255] = {0,}; UINT8 i=0; - UINT8 ret = 0; pbuf[0] = 0x1; pbuf[1] = (UINT8)(opcode); @@ -239,10 +238,7 @@ UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 *p_param_buf) dump(pbuf, param_len+4); - ret = write(fd, pbuf, param_len+4); - if (ret != 0) - return -1; - + write(fd, pbuf, param_len+4); return 0; } @@ -353,7 +349,6 @@ UINT8 DownloadPatchram( char *patchram1 ) INT32 FileSize=0; INT32 SentSize=0; - INT8 ret = 0; DEBUG1( "\n%s\n", patchram1); @@ -390,13 +385,9 @@ UINT8 DownloadPatchram( char *patchram1 ) len = buffer[3]; - ret = fread(&buffer[4],sizeof(UINT8),len, pFile); - if (ret != 0) - return -1; + fread(&buffer[4],sizeof(UINT8),len, pFile); - ret = write(fd, buffer, len + 4); - if (ret != 0) - return -1; + write(fd, buffer, len + 4); /* dispaly progress*/ SentSize += (len + 3); @@ -790,14 +781,14 @@ int main(int argc, char *argv[]) { char text[BTUI_MAX_STRING_LENGTH_PER_LINE]; - if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile))!= NULL) - sscanf(text,"%02x%02x",&bdaddr[0],&bdaddr[1]); + fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + sscanf(text,"%02x%02x",&bdaddr[0],&bdaddr[1]); - if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) != NULL) - sscanf(text,"%02x",&bdaddr[2]); + fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + sscanf(text,"%02x",&bdaddr[2]); - if ((fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) != NULL) - sscanf(text,"%02x%02x%02x",&bdaddr[3],&bdaddr[4],&bdaddr[5]); + fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + sscanf(text,"%02x%02x%02x",&bdaddr[3],&bdaddr[4],&bdaddr[5]); fprintf(stderr,"Writing B/D Address = %02X:%02X:%02X:%02X:%02X:%02X\n",bdaddr[0],bdaddr[1],bdaddr[2],bdaddr[3],bdaddr[4],bdaddr[5]); -- 2.7.4 From 98cbffd1f97848a858aee96a31dc848f0f21e109 Mon Sep 17 00:00:00 2001 From: Taejin Woo Date: Tue, 12 Apr 2016 17:55:07 +0900 Subject: [PATCH 07/16] Add Gear S2 firmware Change-Id: I7156150375ade5050ea806133347e7c0f9dffe15 Signed-off-by: Taejin Woo --- LICENSE.APLv2 | 204 +++++ LICENSE.Broadcom | 64 ++ NOTICE | 10 + firmware/CMakeLists.txt | 4 + ...343A1_001.002.009.0022.0050_Murata_Type-1FR.hcd | Bin 0 -> 19019 bytes ...343A1_001.002.009.0035.0096_ORC_Orbis_WC1-S.hcd | Bin 0 -> 33608 bytes packaging/bluetooth-firmware-bcm.spec | 22 + scripts/CMakeLists.txt | 4 + scripts/bt-dev-start-exynos3250.sh | 119 +++ tools/CMakeLists.txt | 6 + tools/bcmtool_4343w.c | 980 +++++++++++++++++++++ 11 files changed, 1413 insertions(+) create mode 100644 LICENSE.APLv2 create mode 100644 LICENSE.Broadcom create mode 100644 NOTICE create mode 100755 firmware/bcm4343w/BCM4343A1_001.002.009.0022.0050_Murata_Type-1FR.hcd create mode 100644 firmware/bcm4343w/BCM4343A1_001.002.009.0035.0096_ORC_Orbis_WC1-S.hcd create mode 100755 scripts/bt-dev-start-exynos3250.sh create mode 100644 tools/bcmtool_4343w.c diff --git a/LICENSE.APLv2 b/LICENSE.APLv2 new file mode 100644 index 0000000..a06208b --- /dev/null +++ b/LICENSE.APLv2 @@ -0,0 +1,204 @@ +Copyright (c) 2000 - 2011 Samsung Electronics Co., Ltd. All rights reserved. + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + 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. + diff --git a/LICENSE.Broadcom b/LICENSE.Broadcom new file mode 100644 index 0000000..acf4e62 --- /dev/null +++ b/LICENSE.Broadcom @@ -0,0 +1,64 @@ +SOFTWARE LICENSE AGREEMENT + +The accompanying software in binary code form ("Software"), is licensed to you, +or, if you are accepting on behalf of an entity, the entity and its affiliates +exercising rights hereunder ("Licensee") subject to the terms of this software +license agreement ("Agreement"), unless Licensee and Broadcom Corporation +("Broadcom") execute a separate written software license agreement governing +use of the Software. ANY USE, REPRODUCTION, OR DISTRIBUTION OF THE SOFTWARE +CONSTITUTES LICENSEE'S ACCEPTANCE OF THIS AGREEMENT. + +1. License. Subject to the terms and conditions of this Agreement, +Broadcom hereby grants to Licensee a limited, non-exclusive, non-transferable, +royalty-free license: (i) to use and integrate the Software with any other +software; and (ii) to reproduce and distribute the Software complete, +unmodified, and as provided by Broadcom, solely for use with Broadcom +proprietary integrated circuit product(s) sold by Broadcom with which the +Software was designed to be used, or their successors. + +2. Restrictions. Licensee shall distribute Software with a copy of this +Agreement. Licensee shall not remove, efface or obscure any copyright or +trademark notices from the Software. Reproductions of the Broadcom copyright +notice shall be included with each copy of the Software, except where such +Software is embedded in a manner not readily accessible to the end user. +Licensee shall not: (i) use, license, sell or otherwise distribute the Software +except as provided in this Agreement; (ii) attempt to modify in any way, +reverse engineer, decompile or disassemble any portion of the Software; or +(iii) use the Software or other material in violation of any applicable law or +regulation, including but not limited to any regulatory agency. This Agreement +shall automatically terminate upon Licensee’s failure to comply with any of the +terms of this Agreement. In such event, Licensee will destroy all copies of the +Software and its component parts. + +3. Ownership. The Software is licensed and not sold. Title to and +ownership of the Software, including all intellectual property rights thereto, +and any portion thereof remain with Broadcom or its licensors. Licensee hereby +covenants that it will not assert any claim that the Software created by or for +Broadcom infringe any intellectual property right owned or controlled by +Licensee. + +4. Disclaimer. THE SOFTWARE IS OFFERED "AS IS," AND BROADCOM PROVIDES AND +GRANTS AND LICENSEE RECEIVES NO SUPPORT AND NO WARRANTIES OF ANY KIND, EXPRESS +OR IMPLIED, BY STATUTE, COMMUNICATION OR CONDUCT WITH LICENSEE, OR OTHERWISE. +BROADCOM SPECIFICALLY DISCLAIMS ANY IMPLIED WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A SPECIFIC PURPOSE, OR NONINFRINGEMENT CONCERNING THE SOFTWARE OR +ANY UPGRADES TO OR DOCUMENTATION FOR THE SOFTWARE. WITHOUT LIMITATION OF THE +ABOVE, BROADCOM GRANTS NO WARRANTY THAT THE SOFTWARE IS ERROR-FREE OR WILL +OPERATE WITHOUT INTERRUPTION, AND GRANTS NO WARRANTY REGARDING ITS USE OR THE +RESULTS THEREFROM INCLUDING, WITHOUT LIMITATION, ITS CORRECTNESS, ACCURACY, OR +RELIABILITY. TO THE MAXIMUM EXTENT PERMITTED BY LAW, IN NO EVENT SHALL BROADCOM +OR ANY OF ITS LICENSORS HAVE ANY LIABILITY FOR ANY DIRECT, INDIRECT, +INCIDENTAL, SPECIAL, OR CONSEQUENTIAL DAMAGES, HOWEVER CAUSED AND ON ANY THEORY +OF LIABILITY, WHETHER FOR BREACH OF CONTRACT, TORT (INCLUDING NEGLIGENCE) OR +OTHERWISE, ARISING OUT OF THIS AGREEMENT OR USE, REPRODUCTION, OR DISTRIBUTION +OF THE SOFTWARE, INCLUDING BUT NOT LIMITED TO LOSS OF DATA AND LOSS OF PROFITS, +EVEN IF SUCH PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. THESE +LIMITATIONS SHALL APPLY NOTWITHSTANDING ANY FAILURE OF ESSENTIAL PURPOSE OF ANY +LIMITED REMEDY. + +5. Export Laws. LICENSEE UNDERSTANDS AND AGREES THAT THE SOFTWARE IS +SUBJECT TO UNITED STATES AND OTHER APPLICABLE EXPORT-RELATED LAWS AND +REGULATIONS AND THAT LICENSEE MAY NOT EXPORT, RE-EXPORT OR TRANSFER THE +SOFTWARE OR ANY DIRECT PRODUCT OF THE SOFTWARE EXCEPT AS PERMITTED UNDER THOSE +LAWS. WITHOUT LIMITING THE FOREGOING, EXPORT, RE-EXPORT, OR TRANSFER OF THE +SOFTWARE TO CUBA, IRAN, NORTH KOREA, SUDAN, AND SYRIA IS PROHIBITED. diff --git a/NOTICE b/NOTICE new file mode 100644 index 0000000..411d0c9 --- /dev/null +++ b/NOTICE @@ -0,0 +1,10 @@ +Copyright (c) 2012 Samsung Electronics Co., Ltd. All rights reserved. +Except as noted, this software is licensed under Apache License, Version 2. +Please, see the LICENSE.APLv2 file for Apache License terms and conditions. + +The following files are copyrighted by Broadcom and licensed under +a separate license. Please, see the LICENSE.Broadcom file for +license terms and conditions. + +Copyright (c) 2012 Broadcom Co., Ltd. All rights reserved. +- firmware/*.hcd diff --git a/firmware/CMakeLists.txt b/firmware/CMakeLists.txt index 503ae47..b27a0db 100755 --- a/firmware/CMakeLists.txt +++ b/firmware/CMakeLists.txt @@ -1,3 +1,7 @@ # install firmware #INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/BT_FW_BCM4330B1_002.001.003.0221.0265.hcd DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/BT_FW_BCM4358A1_001.002.005.0032.0066.hcd DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) + +# Gear S2 +INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/bcm4343w/BCM4343A1_001.002.009.0035.0096_ORC_Orbis_WC1-S.hcd DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) +INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/bcm4343w/BCM4343A1_001.002.009.0022.0050_Murata_Type-1FR.hcd DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) diff --git a/firmware/bcm4343w/BCM4343A1_001.002.009.0022.0050_Murata_Type-1FR.hcd b/firmware/bcm4343w/BCM4343A1_001.002.009.0022.0050_Murata_Type-1FR.hcd new file mode 100755 index 0000000000000000000000000000000000000000..a85c4a22c0088e4db20b2a6b2f17cd15dfd21ea1 GIT binary patch literal 19019 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 a{yv#Af~z8Xfl4|3V6rO9<#fyYaQ!!Uc70I* literal 0 HcmV?d00001 diff --git a/packaging/bluetooth-firmware-bcm.spec b/packaging/bluetooth-firmware-bcm.spec index 907fc89..a1928b6 100644 --- a/packaging/bluetooth-firmware-bcm.spec +++ b/packaging/bluetooth-firmware-bcm.spec @@ -14,6 +14,12 @@ BuildRequires: cmake %description firmware and tools for bluetooth +%package exynos3250 +Summary: bcm firmware and tools for exynos3250 +Group: TO_BE/FILLED + +%description exynos3250 +bcm firmware and tools for exynos3250 %prep %setup -q @@ -24,12 +30,17 @@ make %{?jobs:-j%jobs} %install rm -rf %{buildroot} + %make_install +mkdir -p %{buildroot}/usr/share/license +cp LICENSE.APLv2 %{buildroot}/usr/share/license/%{name} +cat %{_builddir}/%{name}-%{version}/LICENSE.Broadcom >> %{buildroot}/usr/share/license/%{name} %files %defattr(-,root,root,-) #%{_bindir}/bcmtool_4330b1 +%exclude %{_bindir}/bcmtool_4343w %{_bindir}/bcmtool_4358a1 %{_bindir}/setbd #%{_prefix}/etc/bluetooth/BT_FW_BCM4330B1_002.001.003.0221.0265.hcd @@ -37,3 +48,14 @@ rm -rf %{buildroot} %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-end.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-start.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-set-addr.sh + +%files exynos3250 +%defattr(-,root,root,-) +%{_bindir}/bcmtool_4343w +%{_bindir}/setbd +%{_prefix}/etc/bluetooth/BCM4343A1_001.002.009.0035.0096_ORC_Orbis_WC1-S.hcd +%{_prefix}/etc/bluetooth/BCM4343A1_001.002.009.0022.0050_Murata_Type-1FR.hcd +%attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-end.sh +%attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-start-exynos3250.sh +%attr(755,-,-) %{_prefix}/etc/bluetooth/bt-set-addr.sh +/usr/share/license/%{name} diff --git a/scripts/CMakeLists.txt b/scripts/CMakeLists.txt index 863498d..240b6f0 100755 --- a/scripts/CMakeLists.txt +++ b/scripts/CMakeLists.txt @@ -2,3 +2,7 @@ INSTALL(PROGRAMS ${CMAKE_CURRENT_SOURCE_DIR}/bt-dev-end.sh DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) INSTALL(PROGRAMS ${CMAKE_CURRENT_SOURCE_DIR}/bt-dev-start.sh DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) INSTALL(PROGRAMS ${CMAKE_CURRENT_SOURCE_DIR}/bt-set-addr.sh DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) + +#IF (TIZEN_WEARABLE) +INSTALL(PROGRAMS ${CMAKE_CURRENT_SOURCE_DIR}/bt-dev-start-exynos3250.sh DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) +#ENDIF (TIZEN_WEARABLE) \ No newline at end of file diff --git a/scripts/bt-dev-start-exynos3250.sh b/scripts/bt-dev-start-exynos3250.sh new file mode 100755 index 0000000..555c849 --- /dev/null +++ b/scripts/bt-dev-start-exynos3250.sh @@ -0,0 +1,119 @@ +#!/bin/sh + +GREP="/bin/grep" +MKNOD="/bin/mknod" +AWK="/usr/bin/awk" +RFKILL="/usr/sbin/rfkill" +CP="/bin/cp" + +# +# Script for registering Broadcom UART BT device +# +BT_UART_DEVICE=/dev/ttySAC0 +BT_CHIP_TYPE=bcm2035 +BCM_TOOL=/usr/bin/bcmtool_4343w + +BCM_TOOL_DBG_LOG=/var/lib/bluetooth/bcmtool_log + +# If you want to enable bcmtool debug log, please uncomment below # +#ENABLE_BCMTOOL_DEBUG="-DEBUG" + +HCI_CONFIG=/usr/bin/hciconfig +HCI_ATTACH=/usr/bin/hciattach + +BT_ADDR=/csa/bluetooth/.bd_addr +BCM_WIFI_CID=/opt/etc/.cid.info + +SYSLOG_PATH=/var/log/messages + +UART_SPEED=3000000 + +#Firmware Loading timeout: Unit * 100ms +# Example : 34 is 3.4 sec +TIMEOUT=34 + +HARDWARE=`grep Hardware /proc/cpuinfo | awk "{print \\$3}"` +REVISION=`grep Revision /proc/cpuinfo | awk "{print \\$3}"` +BCM_PACKAGING_TYPE=`cat ${BCM_WIFI_CID}` + +BCM_FIRMWARE="BCM4343A1_001.002.009.0035.0096_ORC_Orbis_WC1-S.hcd" + +if [ "$HARDWARE" == "WC1-S" ] && ( [ "$REVISION" == "0000" ] || [ "$REVISION" == "0001" ] ) +then + BCM_FIRMWARE="BCM4334W0_001.002.003.0014.0017_Ponte_Solo_Semco_B58_13.5dBm.hcd" +else + if [ "${BCM_PACKAGING_TYPE}" == "semco" ] || [ "${BCM_PACKAGING_TYPE}" == "samsung" ] ; then + BCM_FIRMWARE="BCM4343A1_001.002.009.0035.0096_ORC_Orbis_WC1-S.hcd" + echo "Package type is semco" + elif [ "${BCM_PACKAGING_TYPE}" == "murata" ] ; then + BCM_FIRMWARE="BCM4343A1_001.002.009.0022.0050_Murata_Type-1FR.hcd" + echo "Package type is murata" + else + echo "Package type is not detected(${BCM_PACKAGING_TYPE})" + fi +fi + +BCM_CHIP_NAME="BCM4343W" + +echo "Check for Bluetooth device status" +if (${HCI_CONFIG} | grep hci); then + echo "Bluetooth device is UP" + ${HCI_CONFIG} hci0 up + exit 1 +fi + +${RFKILL} unblock bluetooth + +echo "BCM_FIRMWARE: $BCM_FIRMWARE, HARDWARE: $HARDWARE, REVISION: $REVISION" + +# Set BT address: This will internally check for the file presence +/usr/bin/setbd + +#if the setbd return non 0, which means incorrect bd address file, then exit +if [ $? -ne 0 ] +then + exit 1 +fi + +echo "Registering Bluetooth device" + +$BCM_TOOL $BT_UART_DEVICE -TYPE=${BCM_CHIP_NAME} $ENABLE_BCMTOOL_DEBUG \ + -FILE=/usr/etc/bluetooth/$BCM_FIRMWARE -BAUD=$UART_SPEED \ + -ADDR=$BT_ADDR >$BCM_TOOL_DBG_LOG 2>&1 & +bcmtool_pid=$! + +#Check next timeout seconds for bcmtool success +for (( i=1; i<=$TIMEOUT; i++)) +do + sleep 0.1 + kill -0 $bcmtool_pid + bcmtool_alive=$? + + if [ $i -eq $TIMEOUT ] + then + echo "time expired happen $i" + kill -TERM $bcmtool_pid + /usr/sbin/rfkill block bluetooth + ${CP} $SYSLOG_PATH /var/lib/bluetooth/ + exit 1 + fi + + if [ $bcmtool_alive -eq 0 ] + then + echo "Continue....$i" + continue + else + echo "Break.......$i" + break + fi +done + +# Attaching Broadcom device +if (${HCI_ATTACH} $BT_UART_DEVICE -s $UART_SPEED $BT_CHIP_TYPE $UART_SPEED flow); then + /bin/sleep 0.1 + echo "HCIATTACH success" +else + echo "HCIATTACH failed" + ${RFKILL} block bluetooth + ${CP} $SYSLOG_PATH /var/lib/bluetooth/ +fi diff --git a/tools/CMakeLists.txt b/tools/CMakeLists.txt index d37148e..2e7f39e 100644 --- a/tools/CMakeLists.txt +++ b/tools/CMakeLists.txt @@ -4,7 +4,10 @@ PROJECT(bcmtool C) SET(SRCS_4330B1 bcmtool_4330b1.c) SET(BCMTOOL_4330B1 ${PROJECT_NAME}_4330b1) +SET(SRCS_4343W bcmtool_4343w.c) + SET(SRCS_4358A1 bcmtool_4358a1.c) +SET(BCMTOOL_4343W ${PROJECT_NAME}_4343w) SET(BCMTOOL_4358A1 ${PROJECT_NAME}_4358a1) SET(PREFIX ${CMAKE_INSTALL_PREFIX}) @@ -20,10 +23,13 @@ ADD_DEFINITIONS("-DPREFIX=\"${CMAKE_INSTALL_PREFIX}\"") ADD_DEFINITIONS("-DFACTORYFS=\"$ENV{FACTORYFS}\"") ADD_EXECUTABLE(${BCMTOOL_4330B1} ${SRCS_4330B1}) +ADD_EXECUTABLE(${BCMTOOL_4343W} ${SRCS_4343W}) TARGET_LINK_LIBRARIES(${BCMTOOL_4330B1} ${package_LDFLAGS}) ADD_EXECUTABLE(${BCMTOOL_4358A1} ${SRCS_4358A1}) +TARGET_LINK_LIBRARIES(${BCMTOOL_4343W} ${package_LDFLAGS}) TARGET_LINK_LIBRARIES(${BCMTOOL_4358A1} ${package_LDFLAGS}) # install binary file +INSTALL(TARGETS ${BCMTOOL_4343W} DESTINATION ${PLUGIN_INSTALL_PREFIX}/bin) INSTALL(TARGETS ${BCMTOOL_4358A1} DESTINATION ${PLUGIN_INSTALL_PREFIX}/bin) diff --git a/tools/bcmtool_4343w.c b/tools/bcmtool_4343w.c new file mode 100644 index 0000000..1335452 --- /dev/null +++ b/tools/bcmtool_4343w.c @@ -0,0 +1,980 @@ +/***************************************************************************** +** +** Name: bcmtool.c +** +** Description: Download a patchram files for the HCD format +** +** Copyright (c) 2000-2009, Broadcom Corp., All Rights Reserved. +******************************************************************************/ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define N_HCI 15 +#define HCI_UART_H4 0 +#define HCI_UART_BCSP 1 +#define HCI_UART_3WIRE 2 +#define HCI_UART_H4DS 3 +#define HCI_UART_LL 4 +#define HCIUARTSETPROTO _IOW('U', 200, int) +#define HCIUARTGETPROTO _IOR('U', 201, int) +#define HCIUARTGETDEVICE _IOR('U', 202, int) + +/* BT_WAKE Polarity - 0=Active Low, 1= Active High */ +#define HCILP_BT_WAKE_POLARITY 1 + +/* HOST_WAKE Polarity - 0=Active Low, 1= Active High */ +#define HCILP_HOST_WAKE_POLARITY 1 + +/* Local Feature */ +#define BCM_DISABLE_RF_PWRCTRL FALSE + +#define RELEASE_DATE "2011.02.07" +#define DEBUG 1 + +typedef unsigned char UINT8; +typedef unsigned short UINT16; +typedef unsigned long UINT32; +typedef signed long INT32; +typedef signed char INT8; +typedef signed short INT16; +typedef unsigned char BOOLEAN; + +#define FALSE 0 +#define TRUE (!FALSE) + +#define BD_ADDR_LEN 6 /* Device address length */ +typedef UINT8 BD_ADDR[BD_ADDR_LEN]; /* Device address */ + +#define HCI_GRP_LINK_CONTROL_CMDS (0x01 << 10) +#define HCI_GRP_LINK_POLICY_CMDS (0x02 << 10) +#define HCI_GRP_HOST_CONT_BASEBAND_CMDS (0x03 << 10) +#define HCI_GRP_INFORMATIONAL_PARAMS (0x04 << 10) +#define HCI_GRP_STATUS_PARAMS (0x05 << 10) +#define HCI_GRP_TESTING_CMDS (0x06 << 10) +#define HCI_GRP_L2CAP_CMDS (0x07 << 10) +#define HCI_GRP_L2CAP_HCI_EVTS (0x08 << 10) +#define HCI_GRP_VENDOR_SPECIFIC (0x3F << 10) + +#define HCI_RESET (0x0003 | HCI_GRP_HOST_CONT_BASEBAND_CMDS) +#define HCI_SET_EVENT_FILTER (0x0005 | HCI_GRP_HOST_CONT_BASEBAND_CMDS) +#define HCI_READ_LOCAL_NAME (0x0014 | HCI_GRP_HOST_CONT_BASEBAND_CMDS) +#define HCI_READ_SCAN_ENABLE (0x0019 | HCI_GRP_HOST_CONT_BASEBAND_CMDS) +#define HCI_WRITE_SCAN_ENABLE (0x001A | HCI_GRP_HOST_CONT_BASEBAND_CMDS) + +#define HCI_READ_LOCAL_VERSION_INFO (0x0001 | HCI_GRP_INFORMATIONAL_PARAMS) +#define HCI_READ_LOCAL_FEATURES (0x0003 | HCI_GRP_INFORMATIONAL_PARAMS) + +#define HCI_ENABLE_DEV_UNDER_TEST_MODE (0x0003 | HCI_GRP_TESTING_CMDS) + +#define HCI_BRCM_SUPER_PEEK_POKE (0x000A | HCI_GRP_VENDOR_SPECIFIC) +#define VSC_WRITE_BD_ADDR (0x0001 | HCI_GRP_VENDOR_SPECIFIC) +#define VSC_HCI_CMD_SET_LOC_FEATURES_CMD (0x000B | HCI_GRP_VENDOR_SPECIFIC) +#define HCI_BRCM_UPDATE_BAUDRATE_CMD (0x0018 | HCI_GRP_VENDOR_SPECIFIC) +#define HCI_BRCM_WRITE_SCO_PCM_INT_PARAM (0x001C | HCI_GRP_VENDOR_SPECIFIC) +#define VSC_WRITE_PCM_DATA_FORMAT_PARAM (0x001E | HCI_GRP_VENDOR_SPECIFIC) +#define HCI_BRCM_WRITE_SLEEP_MODE (0x0027 | HCI_GRP_VENDOR_SPECIFIC) +#define HCI_BRCM_DOWNLOAD_MINI_DRV (0x002E | HCI_GRP_VENDOR_SPECIFIC) +#define VSC_WRITE_UART_CLOCK_SETTING (0x0045 | HCI_GRP_VENDOR_SPECIFIC) +#define HCI_VSC_WRITE_RAM (0x004C | HCI_GRP_VENDOR_SPECIFIC) +#define HCI_VSC_LAUNCH_RAM (0x004E | HCI_GRP_VENDOR_SPECIFIC) + +#define VOICE_SETTING_MU_LAW_MD 0x0100 +#define VOICE_SETTING_LINEAR_MD 0x0060 + +#define HCI_ARM_MEM_PEEK 0x04 +#define HCI_ARM_MEM_POKE 0x05 + +#define BTUI_MAX_STRING_LENGTH_PER_LINE 255 +#define HCI_BRCM_WRITE_SLEEP_MODE_LENGTH 12 + +#define HCI_BRCM_UPDATE_BAUD_RATE_ENCODED_LENGTH 0x02 +#define HCI_BRCM_UPDATE_BAUD_RATE_UNENCODED_LENGTH 0x06 + +#define VSC_WRITE_UART_CLOCK_SETTING_LEN 1 + +/* print string with time stamp */ +#define TDEBUG0(m) if(debug_mode) {print_time();fprintf(stderr,m);} +#define TDEBUG1(m,n1) if(debug_mode) {print_time();fprintf(stderr,m,n1);} +#define TDEBUG2(m,n1,n2) if(debug_mode) {print_time();fprintf(stderr,m,n1,n2);} +#define TDEBUG3(m,n1,n2,n3) if(debug_mode) {print_time();fprintf(stderr,m,n1,n2,n3);} +#define TDEBUG4(m,n1,n2,n3,n4) if(debug_mode) {print_time();fprintf(stderr,m,n1,n2,n3,n4);} +#define TDEBUG5(m,n1,n2,n3,n4,n5) if(debug_mode) {print_time();fprintf(stderr,m,n1,n2,n3,n4,n5);} +#define TDEBUG6(m,n1,n2,n3,n4,n5,n6) if(debug_mode) {print_time();fprintf(stderr,m,n1,n2,n3,n4,n5,n6);} + +/* print just string */ +#define DEBUG0(m) if(debug_mode) {fprintf(stderr,m);} +#define DEBUG1(m,n1) if(debug_mode) {fprintf(stderr,m,n1);} +#define DEBUG2(m,n1,n2) if(debug_mode) {fprintf(stderr,m,n1,n2);} +#define DEBUG3(m,n1,n2,n3) if(debug_mode) {fprintf(stderr,m,n1,n2,n3);} +#define DEBUG4(m,n1,n2,n3,n4) if(debug_mode) {fprintf(stderr,m,n1,n2,n3,n4);} +#define DEBUG5(m,n1,n2,n3,n4,n5) if(debug_mode) {fprintf(stderr,m,n1,n2,n3,n4,n5);} +#define DEBUG6(m,n1,n2,n3,n4,n5,n6) if(debug_mode) {fprintf(stderr,m,n1,n2,n3,n4,n5,n6);} + +#define STREAM_TO_UINT8(u8, p) {u8 = (UINT8)(*(p)); (p) += 1;} +#define STREAM_TO_UINT16(u16, p) {u16 = ((UINT16)(*(p)) + (((UINT16)(*((p) + 1))) << 8)); (p) += 2;} +#define STREAM_TO_UINT32(u32, p) {u32 = (((UINT32)(*(p))) + ((((UINT32)(*((p) + 1)))) << 8) + ((((UINT32)(*((p) + 2)))) << 16) + ((((UINT32)(*((p) + 3)))) << 24)); (p) += 4;} + +#define ROTATE_BD_ADDR(p1, p2) \ + do \ + { \ + p1[0] = p2[5]; \ + p1[1] = p2[4]; \ + p1[2] = p2[3]; \ + p1[3] = p2[2]; \ + p1[4] = p2[1]; \ + p1[5] = p2[0]; \ + } while (0) + +UINT8 vsc_for_pcm_config[5] = { 0x00, 0x00, 0x03, 0x03, 0x00 }; + +/* + Byte1 -- 0 for MSb first + Byte2 -- 0 Fill value + Byte3 -- 1 Fill option (0:0's, 1:1's , 2:Signed, 3:Programmable) + Byte4 -- 1 Number of fill bits + Byte5 -- 1 Right justified (0 for left justified) +*/ + +UINT8 vsc_for_sco_pcm[5] = { 0x00, 0x01, 0x00, 0x01, 0x01 }; + +/* + Neverland : PCM, 256, short, master ,master + Volance : PCM, 256, short, master ,master + + Byte1 -- 0 for PCM 1 for UART or USB + Byte2 -- 0 : 128, 1: 256, 2:512, 3:1024, 4:2048 Khz + Byte3 -- 0 for short frame sync 1 for long frame sync + Byte4 -- 0 Clock direction 0 for same as sync 1 for opposite direction + Byte5 -- 0 for slave 1 for master +*/ + +int fd; /* HCI handle */ + +BOOLEAN debug_mode = FALSE; /* Debug Mode Enable */ + +BOOLEAN use_two_stop_bits = FALSE; /* Flag of two stop bits for tty */ + +BOOLEAN use_high_speed_download_mode = TRUE; /* Flag of High Speed download mode */ + +unsigned char buffer[1024]; + +typedef enum { + BCM_UNKNOWN = 0, + BCM4330, + BCM4334W, + BCM4343W, +} bcm_product_type; + +bcm_product_type bcm_target_product = BCM_UNKNOWN; + +static struct { + bcm_product_type bcm_product; + char *name; +} bcm_product_table[] = { + { BCM_UNKNOWN, "UNKNOWN"}, + { BCM4330, "BCM4330"}, + { BCM4334W, "BCM4334W"}, + { BCM4343W, "BCM4343W"}, + { 0, NULL } +}; + +struct termios termios; + +void ChangeBaudRate(UINT32 baudrate); + +void exit_err(UINT8 err) +{ + if (use_high_speed_download_mode) + ChangeBaudRate(115200); + + exit(err); +} + +void print_time(void) +{ +#if 0 + struct timespec tp; + int rs; + + rs = clock_gettime(CLOCK_REALTIME, &tp); + fprintf(stderr, "[%04d : %06d]\n", tp.tv_sec, tp.tv_nsec / 1000); + return; +#endif +} + +void dump(unsigned char *out, int len) +{ + int i; + + for (i = 0; i < len; i++) { + if (!(i % 16)) { + DEBUG0("\n"); + } + DEBUG1("%02x ", out[i]); + } + DEBUG0("\n\n"); +} + +UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 * p_param_buf) +{ + UINT8 pbuf[255] = { 0, }; + UINT8 i = 0; + + pbuf[0] = 0x1; + pbuf[1] = (UINT8) (opcode); + pbuf[2] = (UINT8) (opcode >> 8); + pbuf[3] = param_len; + + for (i = 0; i < param_len; i++) { + pbuf[i + 4] = *p_param_buf++; + } + + DEBUG1("Send %d", param_len + 4); + + dump(pbuf, param_len + 4); + + write(fd, pbuf, param_len + 4); + return 0; +} + +void expired(int sig) +{ + static UINT8 count = 0; + DEBUG0("expired try again\n"); + SendCommand(HCI_RESET, 0, NULL); + alarm(1); + count++; + + if (count > 3) { + fprintf(stderr, "[ERR] HCI reset time expired\n"); + exit(1); + } +} + +void read_event(int fd, unsigned char *buffer) +{ + int i = 0; + int len = 3; + int count; + + while ((count = read(fd, &buffer[i], len)) < len) { + i += count; + len -= count; + } + + i += count; + len = buffer[2]; + + while ((count = read(fd, &buffer[i], len)) < len) { + i += count; + len -= count; + } + +#ifdef DEBUG + count += i; + + DEBUG1("\nreceived %d", count); + dump(buffer, count); +#endif +} + +INT32 filesize(char *name) +{ + INT32 size; + int flag; + struct stat buf; + + flag = stat(name, &buf); + if (flag == -1) + return -1; + + size = buf.st_size; + return (size); +} + +void DisplayProgress(int total, int val) +{ +#if 0 +#define PROGRESS_NUM 20 + + int p; + int i; + char text[PROGRESS_NUM + 2] = { 0, }; + + text[0] = '['; + text[PROGRESS_NUM + 1] = ']'; + p = (val * PROGRESS_NUM) / total; + + for (i = 1; i <= p; i++) { + text[i] = '='; + } + + for (i = p + 1; i <= PROGRESS_NUM; i++) { + text[i] = ' '; + } + + for (i = 0; i <= (PROGRESS_NUM + 1); i++) { + fprintf(stderr, "%c", text[i]); + } + + if (p >= PROGRESS_NUM) + fprintf(stderr, " %6d/%6d\n", val, total); + else + fprintf(stderr, " %6d/%6d\r", val, total); +#else + if (val == total) + fprintf(stderr, " %6d/%6d\n", val, total); + else + fprintf(stderr, " %6d/%6d\r", val, total); +#endif +} + +UINT8 DownloadPatchram(char *patchram1) +{ + UINT32 len; + char prm[128] = { 0, }; + FILE *pFile = NULL; + + INT32 FileSize = 0; + INT32 SentSize = 0; + + DEBUG1("\n%s\n", patchram1); + + /* HCI reset */ + DEBUG0("HCI reset\n"); + SendCommand(HCI_RESET, 0, NULL); + alarm(1); + read_event(fd, buffer); + alarm(0); + + if (use_high_speed_download_mode) + ChangeBaudRate(3000000); + + strcpy(prm, patchram1); + + fprintf(stderr, "Download Start\n"); + + if ((pFile = fopen(prm, "r")) == NULL) { + fprintf(stderr, "file %s could not be opened, error %d\n", prm, + errno); + exit_err(1); + } + FileSize = filesize(prm); + + SendCommand(HCI_BRCM_DOWNLOAD_MINI_DRV, 0, NULL); + read_event(fd, buffer); + + usleep(50000); + + while (fread(&buffer[1], sizeof(UINT8), 3, pFile)) { + buffer[0] = 0x01; + + len = buffer[3]; + + fread(&buffer[4], sizeof(UINT8), len, pFile); + + write(fd, buffer, len + 4); + + /* dispaly progress */ + SentSize += (len + 3); +#if 0 + DisplayProgress(FileSize, SentSize); +#endif + /* dispaly progress */ + + read_event(fd, buffer); + + } + fclose(pFile); + + if (bcm_target_product == BCM4343W) { + fprintf(stderr, "Delay 200ms for BCM4343W Wireless Charging Feature\n"); + usleep(200000); /*200ms delay */ + } else { + usleep(100000); /*100ms delay */ + } + + tcflush(fd, TCIOFLUSH); + tcgetattr(fd, &termios); + cfmakeraw(&termios); + termios.c_cflag |= CRTSCTS; + + if (use_two_stop_bits) + termios.c_cflag |= CSTOPB; + + tcsetattr(fd, TCSANOW, &termios); + tcflush(fd, TCIOFLUSH); + tcsetattr(fd, TCSANOW, &termios); + tcflush(fd, TCIOFLUSH); + tcflush(fd, TCIOFLUSH); + cfsetospeed(&termios, B115200); + cfsetispeed(&termios, B115200); + tcsetattr(fd, TCSANOW, &termios); + + /* Send HCI_RESET Command and process event */ + DEBUG0("HCI reset\n"); + SendCommand(HCI_RESET, 0, NULL); + alarm(1); + read_event(fd, buffer); + alarm(0); + fprintf(stderr, "Download Complete\n"); + + return 0; +} + +void SetScanEnable(void) +{ + UINT8 scan_data[1]; + + /* 0x00: No scan enabled */ + /* 0x01: Inquiry scan enabled | Page scan disabled */ + /* 0x02: Inquiry scan disabled | Page scan enabled */ + /* 0x03: Inquiry scan enabled | Page scan enabled */ + + scan_data[0] = 0x03; + SendCommand(HCI_WRITE_SCAN_ENABLE, 1, &scan_data[0]); + read_event(fd, buffer); +} + +/* This patch has been added to write PCM setting for Ponte */ +void SetAudio_for_PCM(void) +{ + fprintf(stderr, "Write Audio parameter for PCM\n"); + + vsc_for_pcm_config[0] = 0x0; + vsc_for_pcm_config[1] = 0x0; + vsc_for_pcm_config[2] = 0x3; /* PCM format 16bit */ + vsc_for_pcm_config[3] = 0x0; + vsc_for_pcm_config[4] = 0x0; + + DEBUG5("vsc_for_pcm_config = {%d,%d,%d,%d,%d}\n", vsc_for_pcm_config[0], + vsc_for_pcm_config[1], vsc_for_pcm_config[2], + vsc_for_pcm_config[3], vsc_for_pcm_config[4]); + + SendCommand(VSC_WRITE_PCM_DATA_FORMAT_PARAM, 5, + (UINT8 *) vsc_for_pcm_config); + read_event(fd, buffer); +} + +void SetAudio(void) +{ + fprintf(stderr, "Write Audio parameter\n"); + + DEBUG5("vsc_for_sco_pcm = {%d,%d,%d,%d,%d}\n", vsc_for_sco_pcm[0], + vsc_for_sco_pcm[1], vsc_for_sco_pcm[2], + vsc_for_sco_pcm[3], vsc_for_sco_pcm[4]); + + SendCommand(HCI_BRCM_WRITE_SCO_PCM_INT_PARAM, 5, + (UINT8 *) vsc_for_sco_pcm); + read_event(fd, buffer); + + DEBUG5("vsc_for_pcm_config = {%d,%d,%d,%d,%d}\n", vsc_for_pcm_config[0], + vsc_for_pcm_config[1], vsc_for_pcm_config[2], + vsc_for_pcm_config[3], vsc_for_pcm_config[4]); + + SendCommand(VSC_WRITE_PCM_DATA_FORMAT_PARAM, 5, + (UINT8 *) vsc_for_pcm_config); + read_event(fd, buffer); +} + +void SetPcmConf(UINT8 p0, UINT8 p1, UINT8 p2, UINT8 p3, UINT8 p4) +{ + vsc_for_pcm_config[0] = p0; + vsc_for_pcm_config[1] = p1; + vsc_for_pcm_config[2] = p2; + vsc_for_pcm_config[3] = p3; + vsc_for_pcm_config[4] = p4; +} + +void SetScoConf(UINT8 p0, UINT8 p1, UINT8 p2, UINT8 p3, UINT8 p4) +{ + vsc_for_sco_pcm[0] = p0; + vsc_for_sco_pcm[1] = p1; + vsc_for_sco_pcm[2] = p2; + vsc_for_sco_pcm[3] = p3; + vsc_for_sco_pcm[4] = p4; +} + +void HCILP_Enable(BOOLEAN on) +{ + /* Host Stack Idle Threshold */ + UINT8 hcilp_idle_threshold = 0x01; + + /* Host Controller Idle Threshold */ + UINT8 hcilp_hc_idle_threshold = 0x01; + + if (bcm_target_product == BCM4343W) { + hcilp_idle_threshold = 0x0A; + hcilp_hc_idle_threshold = 0x0A; + } + + fprintf(stderr, "Set Low Power mode %d for %s\n", on, + bcm_product_table[bcm_target_product].name); + + UINT8 data[HCI_BRCM_WRITE_SLEEP_MODE_LENGTH] = { + 0x01, /* Sleep Mode algorithm 1 */ + hcilp_idle_threshold, /* Host Idle Treshold */ + hcilp_hc_idle_threshold, /* Host Controller Idle Treshold*//* this should be less than scan interval. */ + HCILP_BT_WAKE_POLARITY, /* BT_WAKE Polarity - 0=Active Low, 1= Active High */ + HCILP_HOST_WAKE_POLARITY, /* HOST_WAKE Polarity - 0=Active Low, 1= Active High */ + 0x01, /* Allow host Sleep during SCO */ + 0x01, /* Combine Sleep Mode and LPM - The device will not sleep in mode 0 if this flag is set to 1, */ + 0x00, /* UART_TXD Tri-State : 0x00 = Do not tri-state UART_TXD in sleep mode */ + 0x00, /* NA to Mode 1 */ + 0x00, /* NA to Mode 1 */ + 0x00, /* NA*/ + 0x00, /* NA*/ + }; + + if (on) { + data[0] = 0x01; + } else { + data[0] = 0x00; + } + + SendCommand(HCI_BRCM_WRITE_SLEEP_MODE, HCI_BRCM_WRITE_SLEEP_MODE_LENGTH, + (UINT8 *) data); + read_event(fd, buffer); +} + +UINT32 uart_speed(UINT32 Speed) +{ + switch (Speed) { + case 115200: + return B115200; + case 230400: + return B230400; + case 460800: + return B460800; + case 921600: + return B921600; + case 1000000: + return B1000000; + case 1500000: + return B1500000; + case 2000000: + return B2000000; + case 2500000: + return B2500000; + case 3000000: + return B3000000; + case 4000000: + return B4000000; + default: + return B115200; + } +} + +void ChangeBaudRate(UINT32 baudrate) +{ + UINT8 hci_data[HCI_BRCM_UPDATE_BAUD_RATE_UNENCODED_LENGTH] = + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }; + UINT8 uart_clock_24 = 0x2; /* 0x1 - UART Clock 48MHz, 0x2 - UART Clock 24MHz */ + UINT8 uart_clock_48 = 0x1; /* 0x1 - UART Clock 48MHz, 0x2 - UART Clock 24MHz */ + + switch (baudrate) { + case 115200: + case 230400: + case 460800: + case 921600: + case 1000000: + case 1500000: + case 2000000: + case 2500000: + /* Write UART Clock setting of 24MHz */ + DEBUG0("Change UART_CLOCK 24Mhz\n"); + SendCommand(VSC_WRITE_UART_CLOCK_SETTING, + VSC_WRITE_UART_CLOCK_SETTING_LEN, + (UINT8 *) & uart_clock_24); + read_event(fd, buffer); + break; + + case 3000000: + case 4000000: + /* Write UART Clock setting of 48MHz */ + DEBUG0("Change UART_CLOCK 48Mhz\n"); + SendCommand(VSC_WRITE_UART_CLOCK_SETTING, + VSC_WRITE_UART_CLOCK_SETTING_LEN, + (UINT8 *) & uart_clock_48); + read_event(fd, buffer); + break; + + default: + fprintf(stderr, "Not Support baudrate = %ld\n", baudrate); + exit_err(1); + break; + } + + hci_data[2] = baudrate & 0xFF; + hci_data[3] = (baudrate >> 8) & 0xFF; + hci_data[4] = (baudrate >> 16) & 0xFF; + hci_data[5] = (baudrate >> 24) & 0xFF; + + DEBUG1("Change Baudrate %ld\n", baudrate); + + SendCommand(HCI_BRCM_UPDATE_BAUDRATE_CMD, + HCI_BRCM_UPDATE_BAUD_RATE_UNENCODED_LENGTH, + (UINT8 *) hci_data); + read_event(fd, buffer); + + tcflush(fd, TCIOFLUSH); + tcgetattr(fd, &termios); + cfmakeraw(&termios); + termios.c_cflag |= CRTSCTS; + + if (use_two_stop_bits) + termios.c_cflag |= CSTOPB; + + tcsetattr(fd, TCSANOW, &termios); + tcflush(fd, TCIOFLUSH); + tcsetattr(fd, TCSANOW, &termios); + tcflush(fd, TCIOFLUSH); + tcflush(fd, TCIOFLUSH); + cfsetospeed(&termios, uart_speed(baudrate)); + cfsetispeed(&termios, uart_speed(baudrate)); + tcsetattr(fd, TCSANOW, &termios); + +} + +void EnableTestMode(void) +{ + UINT8 filter_data[] = { 0x02, 0x00, 0x02 }; + + /* bt sleep disable */ + HCILP_Enable(FALSE); + + /* Enable both Inquiry & Page Scans */ + SetScanEnable(); + + /* Set Event Filter: Enable Auto Connect */ + SendCommand(HCI_SET_EVENT_FILTER, 0x03, (UINT8 *) filter_data); + read_event(fd, buffer); + + /* Enable Device under test */ + SendCommand(HCI_ENABLE_DEV_UNDER_TEST_MODE, 0x0, NULL); + read_event(fd, buffer); + + fprintf(stderr, "Enable Device Under Test\n"); +} + +void GetLocalName(void) +{ + UINT8 *data = NULL; + + /* HCI reset */ + DEBUG0("HCI reset\n"); + SendCommand(HCI_RESET, 0, NULL); + alarm(1); + read_event(fd, buffer); + alarm(0); + + DEBUG0("Read Local Name\n"); + SendCommand(HCI_READ_LOCAL_NAME, 0, NULL); + read_event(fd, buffer); + + data = &buffer[7]; + + fprintf(stderr, "Chip Name is %s\n", data); +} + +void SetLocalFeatures(void) +{ + UINT8 *data = NULL; + + DEBUG0("Read Local Feature\n"); + SendCommand(HCI_READ_LOCAL_FEATURES, 0, NULL); + read_event(fd, buffer); + + data = &buffer[7]; + +#if (BCM_DISABLE_RF_PWRCTRL == TRUE) + fprintf(stderr, "Remove Power Control\n"); + data[2] &= 0xFB; /* Power contrel */ +#endif + DEBUG0("Write Local Feature\n"); + SendCommand(VSC_HCI_CMD_SET_LOC_FEATURES_CMD, 0x08, (UINT8 *) data); + read_event(fd, buffer); +} + +void EnbleHCI(void) +{ + int i = N_HCI; + int proto = HCI_UART_H4; + + if (ioctl(fd, TIOCSETD, &i) < 0) { + fprintf(stderr, "Can't set line discipline\n"); + return; + } + + if (ioctl(fd, HCIUARTSETPROTO, proto) < 0) { + fprintf(stderr, "Can't set hci protocol\n"); + return; + } + fprintf(stderr, "Done setting line discpline\n"); + return; + +} + +void SetBcmProductType(char *bcm_product_name) +{ + int i = 0; + + if (bcm_product_name == NULL) { + bcm_target_product = BCM_UNKNOWN; + return; + } + + for (i = 0; bcm_product_table[i].name != NULL; i++) { + if (!strcasecmp(bcm_product_table[i].name,bcm_product_name)) { + bcm_target_product = bcm_product_table[i].bcm_product; + fprintf(stderr, "Detected name is %s\n", + bcm_product_table[i].name); + } + } + + return; +} + +void print_usage(void) +{ + fprintf(stderr, "\n"); + fprintf(stderr, "BRCM BT tool for Linux release %s\n", RELEASE_DATE); + fprintf(stderr, "\n"); + fprintf(stderr, + " Usage: bcmtool [command parameter],....\n\n"); + fprintf(stderr, + " -FILE Patchram file name EX) -FILE=BCM43xx_xxx.hcd\n"); + fprintf(stderr, + " -BAUD Set Baudrate EX) -BAUD=3000000\n"); + fprintf(stderr, + " -ADDR BD addr file name EX) -ADDR=.bdaddr\n"); + fprintf(stderr, " -SCO Enable SCO/PCM config EX) -SCO\n"); + fprintf(stderr, " -PCM_SETTING Write PCM config EX) -PCM_SETTING\n"); + fprintf(stderr, + " -SETSCO SCO/PCM values verify EX) -SETSCO=0,1,0,1,1,0,0,3,3,0\n"); + fprintf(stderr, " -LP Enable Low power EX) -LP\n"); + fprintf(stderr, " -FEATURE Set local Feature EX) -FEATURE\n"); + fprintf(stderr, " -GETNAME Get local Name EX) -GETNAME\n"); + fprintf(stderr, + " -DUT Enable DUT mode(do not use with -LP) EX) -DUT\n"); + fprintf(stderr, + " -ATTACH Attach BT controller to BlueZ stack EX) -ATTACH\n"); + fprintf(stderr, " -DEBUG Debug message EX) -DEBUG\n"); + fprintf(stderr, " -CSTOPB Set two stop bits for tty EX) -CSTOPB\n"); + fprintf(stderr, " -SLOWDOWN Set low speed download mode \n"); + fprintf(stderr, " default is High speed mode EX) -SLOWDOWN\n"); + fprintf(stderr, " -TYPE BCM Product name EX) -TYPE=BCM4343W\n"); + + fprintf(stderr, "\n"); +} + +int main(int argc, char *argv[]) +{ + UINT8 i = 0; + + if (argc < 2) { + print_usage(); + exit(1); + } else { + fprintf(stderr, "BRCM BT tool for Linux release %s\n", + RELEASE_DATE); + } + + /* Open dev port */ + if ((fd = open(argv[1], O_RDWR | O_NOCTTY)) == -1) { + fprintf(stderr, "port %s could not be opened, error %d\n", + argv[1], errno); + exit(2); + } + + tcflush(fd, TCIOFLUSH); + tcgetattr(fd, &termios); + cfmakeraw(&termios); + termios.c_cflag |= CRTSCTS; + + if (use_two_stop_bits) + termios.c_cflag |= CSTOPB; + + tcsetattr(fd, TCSANOW, &termios); + tcflush(fd, TCIOFLUSH); + tcsetattr(fd, TCSANOW, &termios); + tcflush(fd, TCIOFLUSH); + tcflush(fd, TCIOFLUSH); + cfsetospeed(&termios, B115200); + cfsetispeed(&termios, B115200); + tcsetattr(fd, TCSANOW, &termios); + + signal(SIGALRM, expired); + + for (i = 2; i < argc; i++) { + char *ptr = argv[i]; + + if (strstr(ptr, "-DEBUG")) { + debug_mode = TRUE; + DEBUG0("DEBUG On\n"); + + } else if (strstr(ptr, "-CSTOPB")) { + use_two_stop_bits = TRUE; + DEBUG0("Use two stop bits for tty\n"); + + } else if (strstr(ptr, "-SLOWDOWN")) { + use_high_speed_download_mode = FALSE; + DEBUG0("Disable High Speed Download mode\n"); + + } else if (strstr(ptr, "-TYPE=")) { + char bcm_product_name[128]; + ptr +=6; + + strncpy(bcm_product_name,ptr,8); + + SetBcmProductType(bcm_product_name); + } + } + + for (i = 2; i < argc; i++) { + char *ptr = argv[i]; + + if (ptr == NULL) + continue; + + fprintf(stderr, "[%d] %s\n", i - 1, ptr); + + if (strstr(ptr, "-FILE=")) { + char prm_name[128]; + + ptr += 6; + + strncpy(prm_name, ptr, 127); + DownloadPatchram(prm_name); + + } else if (strstr(ptr, "-BAUD=")) { + UINT32 baudrate; + + ptr += 6; + baudrate = atoi(ptr); + + ChangeBaudRate(baudrate); + } else if (strstr(ptr, "-ADDR=")) { + char *bdaddr_filename; + FILE *pFile = NULL; + + int bdaddr[10]; /* Displayed BD Address */ + + BD_ADDR local_addr; /* BD Address for write */ + +#if 0 + ptr += 6; + if (sscanf + (ptr, "%02X:%02X:%02X:%02X:%02X:%02X", &bdaddr[0], + &bdaddr[1], &bdaddr[2], &bdaddr[3], &bdaddr[4], + &bdaddr[5]) != 6) { + fprintf(stderr, "-ADDR: Parameter error"); + exit_err(1); + } + bte_write_bdaddr(bdaddr); +#endif + ptr += 6; + bdaddr_filename = ptr; + + if (bdaddr_filename) { + pFile = fopen(bdaddr_filename, "r"); + } + + if (pFile) { + char text[BTUI_MAX_STRING_LENGTH_PER_LINE]; + + fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, + pFile); + sscanf(text, "%02x%02x", &bdaddr[0], + &bdaddr[1]); + + fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, + pFile); + sscanf(text, "%02x", &bdaddr[2]); + + fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, + pFile); + sscanf(text, "%02x%02x%02x", &bdaddr[3], + &bdaddr[4], &bdaddr[5]); + + fprintf(stderr, + "Writing B/D Address = %02X:%02X:%02X:%02X:%02X:%02X\n", + bdaddr[0], bdaddr[1], bdaddr[2], + bdaddr[3], bdaddr[4], bdaddr[5]); + + ROTATE_BD_ADDR(local_addr, bdaddr); + + SendCommand(VSC_WRITE_BD_ADDR, BD_ADDR_LEN, + (UINT8 *) local_addr); + read_event(fd, buffer); + } else { + fprintf(stderr, "-ADDR: file open fail\n"); + exit_err(1); + } + } else if (strstr(ptr, "-PCM_SETTING")) { + SetAudio_for_PCM(); + + } else if (strstr(ptr, "-SCO")) { + SetAudio(); + + } else if (strstr(ptr, "-SETSCO=")) { + ptr += 8; + int value[10]; + + if (sscanf + (ptr, "%d,%d,%d,%d,%d,%d,%d,%d,%d,%d", &value[0], + &value[1], &value[2], &value[3], &value[4], + &value[5], &value[6], &value[7], &value[8], + &value[9]) != 10) { + DEBUG0("PCM / SCO configuration value err\n"); + DEBUG0 + ("SCO_Routing,PCM_Interface_Rate,Frame_Type,Sync_Mode,Clock_Mode,LSB_First,Fill_bits,Fill_Method,Fill_Num,Right_Justify\n"); + exit_err(1); + } + + SetScoConf(value[0], value[1], value[2], value[3], + value[4]); + SetPcmConf(value[5], value[6], value[7], value[8], + value[9]); + SetAudio(); + } else if (strstr(ptr, "-LP")) { + HCILP_Enable(TRUE); + } else if (strstr(ptr, "-DUT")) { + EnableTestMode(); + } else if (strstr(ptr, "-FEATURE")) { + SetLocalFeatures(); + } else if (strstr(ptr, "-GETNAME")) { + GetLocalName(); + } else if (strstr(ptr, "-ATTACH")) { + EnbleHCI(); + while (1) { + sleep(UINT_MAX); + } + } else if (strstr(ptr, "-DEBUG")) { + } else if (strstr(ptr, "-CSTOPB")) { + } else if (strstr(ptr, "-SLOWDOWN")) { + } else if (strstr(ptr, "-TYPE")) { + } else { + fprintf(stderr, "Invalid parameter(s)!\n"); + exit_err(1); + } + } + + fprintf(stderr, "EXIT\n"); + close(fd); + exit(0); + + return 0; +} -- 2.7.4 From defd059a2156f198f68d134b46d975ab5f8fc363 Mon Sep 17 00:00:00 2001 From: Taejin Woo Date: Wed, 20 Apr 2016 09:14:03 +0900 Subject: [PATCH 08/16] Add the firmware and broadcom's tool for Artik Change-Id: I27e107a261aed30a394ccd8838540845a2e0d5b0 Signed-off-by: Taejin Woo --- ...354_003.001.012.0353.0745_Samsung_Artik_ORC.hcd | Bin 0 -> 83184 bytes firmware/CMakeLists.txt | 2 + firmware/artik_bcmtool/brcm_patchram_plus | Bin 0 -> 14596 bytes packaging/bluetooth-firmware-bcm.spec | 23 +++- scripts/CMakeLists.txt | 3 +- scripts/bt-dev-start-artik.sh | 149 +++++++++++++++++++++ 6 files changed, 175 insertions(+), 2 deletions(-) create mode 100644 firmware/BCM4354_003.001.012.0353.0745_Samsung_Artik_ORC.hcd create mode 100755 firmware/artik_bcmtool/brcm_patchram_plus create mode 100644 scripts/bt-dev-start-artik.sh diff --git a/firmware/BCM4354_003.001.012.0353.0745_Samsung_Artik_ORC.hcd b/firmware/BCM4354_003.001.012.0353.0745_Samsung_Artik_ORC.hcd new file mode 100644 index 0000000000000000000000000000000000000000..1a7e626d28319d83bdaa9821f455721b7d82be06 GIT binary patch literal 83184 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¨znw}I)35=IO@F#_IQ)Hocyt9jkO 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 literal 0 HcmV?d00001 diff --git a/firmware/CMakeLists.txt b/firmware/CMakeLists.txt index b27a0db..aa6eb02 100755 --- a/firmware/CMakeLists.txt +++ b/firmware/CMakeLists.txt @@ -1,6 +1,8 @@ # install firmware #INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/BT_FW_BCM4330B1_002.001.003.0221.0265.hcd DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/BT_FW_BCM4358A1_001.002.005.0032.0066.hcd DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) +INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/BCM4354_003.001.012.0353.0745_Samsung_Artik_ORC.hcd DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) +INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/artik_bcmtool/brcm_patchram_plus DESTINATION ${PLUGIN_INSTALL_PREFIX}/bin) # Gear S2 INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/bcm4343w/BCM4343A1_001.002.009.0035.0096_ORC_Orbis_WC1-S.hcd DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) diff --git a/firmware/artik_bcmtool/brcm_patchram_plus b/firmware/artik_bcmtool/brcm_patchram_plus new file mode 100755 index 0000000000000000000000000000000000000000..0a0871d7b95dcff07020e8bcf870647389507d5f GIT binary patch literal 14596 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 X7J4EPN>N>_xl`*S&`Z}Cs^otILwuBD literal 0 HcmV?d00001 diff --git a/packaging/bluetooth-firmware-bcm.spec b/packaging/bluetooth-firmware-bcm.spec index a1928b6..0a35398 100644 --- a/packaging/bluetooth-firmware-bcm.spec +++ b/packaging/bluetooth-firmware-bcm.spec @@ -16,11 +16,18 @@ BuildRequires: cmake %package exynos3250 Summary: bcm firmware and tools for exynos3250 -Group: TO_BE/FILLED +Group: Hardware Support/Handset %description exynos3250 bcm firmware and tools for exynos3250 +%package artik +Summary: bcm firmware and tools for artik +Group: Hardware Support/Handset + +%description artik +bcm firmware and tools for artik + %prep %setup -q @@ -59,3 +66,17 @@ cat %{_builddir}/%{name}-%{version}/LICENSE.Broadcom >> %{buildroot}/usr/share/l %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-start-exynos3250.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-set-addr.sh /usr/share/license/%{name} + +%post artik +rm -rf %{_prefix}/etc/bluetooth/bt-dev-start.sh +ln -s %{_prefix}/etc/bluetooth/bt-dev-start-artik.sh %{_prefix}/etc/bluetooth/bt-dev-start.sh + +%files artik +%defattr(644,root,root,-) +%{_bindir}/brcm_patchram_plus +%{_bindir}/setbd +%{_prefix}/etc/bluetooth/BCM4354_003.001.012.0353.0745_Samsung_Artik_ORC.hcd +%attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-end.sh +%attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-start-artik.sh +%attr(755,-,-) %{_prefix}/etc/bluetooth/bt-set-addr.sh +/usr/share/license/%{name} diff --git a/scripts/CMakeLists.txt b/scripts/CMakeLists.txt index 240b6f0..d14dac4 100755 --- a/scripts/CMakeLists.txt +++ b/scripts/CMakeLists.txt @@ -5,4 +5,5 @@ INSTALL(PROGRAMS ${CMAKE_CURRENT_SOURCE_DIR}/bt-set-addr.sh DESTINATION ${PLUGIN #IF (TIZEN_WEARABLE) INSTALL(PROGRAMS ${CMAKE_CURRENT_SOURCE_DIR}/bt-dev-start-exynos3250.sh DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) -#ENDIF (TIZEN_WEARABLE) \ No newline at end of file +INSTALL(PROGRAMS ${CMAKE_CURRENT_SOURCE_DIR}/bt-dev-start-artik.sh DESTINATION ${PLUGIN_INSTALL_PREFIX}/etc/bluetooth) +#ENDIF (TIZEN_WEARABLE) diff --git a/scripts/bt-dev-start-artik.sh b/scripts/bt-dev-start-artik.sh new file mode 100644 index 0000000..000860a --- /dev/null +++ b/scripts/bt-dev-start-artik.sh @@ -0,0 +1,149 @@ +#!/bin/sh + +# Script for registering Broadcom UART BT device +BT_UART_DEVICE=/dev/ttySAC3 +BT_CHIP_TYPE=bcm2035 +BCM_TOOL=/usr/bin/brcm_patchram_plus + +BT_PLATFORM_DEFAULT_HCI_NAME="TIZEN-Mobile" +UART_SPEED=3000000 +TIMEOUT=24 + +#set default firmware +BCM_FIRMWARE=BT_FW_BCM4358A1_001.002.005.0032.0066.hcd + +REVISION_NUM=`grep Revision /proc/cpuinfo | awk "{print \\$3}"` +REVISION_HIGH=`echo $REVISION_NUM| cut -c1-2` +REVISION_LOW=`echo $REVISION_NUM| cut -c3-` + +HARDWARE=`grep Hardware /proc/cpuinfo | awk "{print \\$3}"` + +check_hw_dt() +{ + HARDWARE=`cat /proc/device-tree/model | awk "{print \\$2}"` + + case $HARDWARE in + + "ARTIK10") + BT_UART_DEVICE=/dev/ttySAC2 + BCM_TOOL=/usr/bin/brcm_patchram_plus + BCM_FIRMWARE=BCM4354_003.001.012.0353.0745_Samsung_Artik_ORC.hcd + ;; + + "ARTIK5") + BT_UART_DEVICE=/dev/ttySAC0 + BCM_TOOL=/usr/bin/brcm_patchram_plus + BCM_FIRMWARE=BCM4354_003.001.012.0353.0745_Samsung_Artik_ORC.hcd + ;; + esac +} + +parse_bd_addr() +{ + for x in $(cat /proc/cmdline); do + case $x in + bd_addr=*) + BD_ADDR=${x#bd_addr=} + ;; + esac + done +} + +check_hw_dt +if [ -z "${HARDWARE##ARTIK*}" ]; then + parse_bd_addr +fi + +if [ ! -e "$BT_UART_DEVICE" ] +then + mknod $BT_UART_DEVICE c 204 64 +fi + +if [ ! -e /opt/etc/.bd_addr ] +then + # Set BT address + /usr/bin/setbd +fi + +# Trun-on Bluetooth Chip +/usr/sbin/rfkill unblock bluetooth + +echo "Check for Bluetooth device status" +if (/usr/bin/hciconfig | grep hci); then + echo "Bluetooth device is UP" + /usr/bin/hciconfig hci0 up +else + echo "Bluetooth device is DOWN" + echo "Registering Bluetooth device" + echo "change auth of brodcom tool" + chmod 755 $BCM_TOOL + + # In tizenW hardware first time bcmtool download may not success, hence we need to try more times + MAXBCMTOOLTRY=5 + flag=0 + for (( c=1; c<=$MAXBCMTOOLTRY; c++)) + do + echo "******* Bcmtool download attempt $c ********" + + $BCM_TOOL $BT_UART_DEVICE --patchram /usr/etc/bluetooth/$BCM_FIRMWARE --no2bytes --baudrate $UART_SPEED --use_baudrate_for_download $BT_UART_DEVICE --bd_addr ${BD_ADDR} > /dev/null 2>&1 & + bcmtool_pid=$! + #Check next timeout seconds for bcmtool success + for (( i=1; i<=$TIMEOUT; i++)) + do + /bin/sleep 0.1 + kill -0 $bcmtool_pid + bcmtool_alive=$? + + if [ $i -eq $TIMEOUT ] + then + echo "time expired happen $i" + kill -TERM $bcmtool_pid + break +# ${RFKILL} block bluetooth +# exit 1 + fi + + if [ $bcmtool_alive -eq 0 ] + then + echo "Continue....$i" + continue + else + echo "Break.......$i" + flag=1 + break + fi + done + + if [ $flag -eq 1 ] + then + echo "Break bcmtool download loop on $c attempt" + break + else + /bin/sleep 1 + echo "sleep done" + fi + + + if [ $c -eq $MAXBCMTOOLTRY ] + then + echo "***** No Chance to activate, count=$c ******" + ${RFKILL} block bluetooth + exit 1 + fi + + done + + echo "Try for hciattach" + + # Attaching Broadcom device + if (/usr/bin/hciattach $BT_UART_DEVICE -s $UART_SPEED $BT_CHIP_TYPE $UART_SPEED flow); then + sleep 0.1 + /usr/bin/hciconfig hci0 up + /usr/bin/hciconfig hci0 name $BT_PLATFORM_DEFAULT_HCI_NAME + /usr/bin/hciconfig hci0 sspmode 1 + echo "HCIATTACH success" + else + echo "HCIATTACH failed" + /usr/sbin/rfkill block bluetooth + fi +fi -- 2.7.4 From da5e9bf23a68598a0fc506f6fdd470ca8dbcd39c Mon Sep 17 00:00:00 2001 From: DoHyun Pyun Date: Fri, 20 May 2016 12:57:31 +0900 Subject: [PATCH 09/16] Fix the build warning errors Change-Id: Ia164668240cced624f2cf2ef4b722e48e63595c8 Signed-off-by: DoHyun Pyun --- CMakeLists.txt | 2 +- set-address/setbd.c | 6 +++++- tools/bcmtool_4330b1.c | 20 ++++++++++++++------ tools/bcmtool_4343w.c | 33 ++++++++++++++++++++++++--------- tools/bcmtool_4358a1.c | 20 ++++++++++++++------ 5 files changed, 58 insertions(+), 23 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index c0285fa..919624a 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -7,7 +7,7 @@ FOREACH(flag ${package_CFLAGS}) SET(EXTRA_CFLAGS "${EXTRA_CFLAGS} ${flag}") ENDFOREACH(flag) -SET(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${EXTRA_CFLAGS}") +SET(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${EXTRA_CFLAGS} -fPIC -Wall -Werror") ADD_SUBDIRECTORY(set-address) ADD_SUBDIRECTORY(tools) diff --git a/set-address/setbd.c b/set-address/setbd.c index 013a92f..49f9525 100644 --- a/set-address/setbd.c +++ b/set-address/setbd.c @@ -87,6 +87,7 @@ int addremoveBD(char* path, char* pskey){ } ret = fputs(pskey,new); + APP_DBG("fputs ret: %d", ret); while(1){ result = fgets(cmp, READ_BD_FILE_MAX, fd); @@ -106,6 +107,7 @@ int addremoveBD(char* path, char* pskey){ } ret = fputs(cmp,new); + APP_DBG("fputs ret: %d", ret); } return 0; @@ -202,10 +204,12 @@ int make_bt_address(gboolean overwrite_bt_address) #if defined(BT_CHIP_CSR) || defined(BT_CHIP_BROADCOM) int fd; - int i; unsigned char txt[BD_ADDR_LEN]; unsigned char nap[4+1], uap[2+1], lap[6+1]; +#if defined(BT_CHIP_CSR) + int i; char pskey[PSKEY_LEN+3]; +#endif int ret; fd=open(BD_ADDR_FILE, O_RDONLY | O_SYNC); diff --git a/tools/bcmtool_4330b1.c b/tools/bcmtool_4330b1.c index 6350a06..e9947f6 100644 --- a/tools/bcmtool_4330b1.c +++ b/tools/bcmtool_4330b1.c @@ -238,7 +238,9 @@ UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 *p_param_buf) dump(pbuf, param_len+4); - write(fd, pbuf, param_len+4); + if (write(fd, pbuf, param_len + 4) < 0) + DEBUG0("Fail to write pbuf"); + return 0; } @@ -346,6 +348,7 @@ UINT8 DownloadPatchram( char *patchram1 ) UINT32 len; char prm[128] ={0,}; FILE* pFile = NULL; + size_t size; INT32 FileSize=0; INT32 SentSize=0; @@ -385,9 +388,11 @@ UINT8 DownloadPatchram( char *patchram1 ) len = buffer[3]; - fread(&buffer[4],sizeof(UINT8),len, pFile); + size = fread(&buffer[4], sizeof(UINT8), len, pFile); + fprintf(stderr, "fread size: %d\n", size); - write(fd, buffer, len + 4); + size = write(fd, buffer, len + 4); + fprintf(stderr, "write size: %d\n", size); /* dispaly progress*/ SentSize += (len + 3); @@ -781,13 +786,16 @@ int main(int argc, char *argv[]) { char text[BTUI_MAX_STRING_LENGTH_PER_LINE]; - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + if (!fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) + fprintf(stderr, "fail to fgets"); sscanf(text,"%02x%02x",&bdaddr[0],&bdaddr[1]); - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + if (!fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) + fprintf(stderr, "fail to fgets"); sscanf(text,"%02x",&bdaddr[2]); - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + if (!fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) + fprintf(stderr, "fail to fgets"); sscanf(text,"%02x%02x%02x",&bdaddr[3],&bdaddr[4],&bdaddr[5]); fprintf(stderr,"Writing B/D Address = %02X:%02X:%02X:%02X:%02X:%02X\n",bdaddr[0],bdaddr[1],bdaddr[2],bdaddr[3],bdaddr[4],bdaddr[5]); diff --git a/tools/bcmtool_4343w.c b/tools/bcmtool_4343w.c index 1335452..4fdbfc4 100644 --- a/tools/bcmtool_4343w.c +++ b/tools/bcmtool_4343w.c @@ -245,7 +245,9 @@ UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 * p_param_buf) dump(pbuf, param_len + 4); - write(fd, pbuf, param_len + 4); + if (write(fd, pbuf, param_len + 4) < 0) + DEBUG0("Fail to write pbuf"); + return 0; } @@ -346,8 +348,11 @@ UINT8 DownloadPatchram(char *patchram1) UINT32 len; char prm[128] = { 0, }; FILE *pFile = NULL; + size_t size; +#if 0 INT32 FileSize = 0; +#endif INT32 SentSize = 0; DEBUG1("\n%s\n", patchram1); @@ -371,7 +376,10 @@ UINT8 DownloadPatchram(char *patchram1) errno); exit_err(1); } + +#if 0 FileSize = filesize(prm); +#endif SendCommand(HCI_BRCM_DOWNLOAD_MINI_DRV, 0, NULL); read_event(fd, buffer); @@ -383,9 +391,11 @@ UINT8 DownloadPatchram(char *patchram1) len = buffer[3]; - fread(&buffer[4], sizeof(UINT8), len, pFile); + size = fread(&buffer[4], sizeof(UINT8), len, pFile); + fprintf(stderr, "fread size: %d\n", size); - write(fd, buffer, len + 4); + size = write(fd, buffer, len + 4); + fprintf(stderr, "write size: %d\n", size); /* dispaly progress */ SentSize += (len + 3); @@ -893,19 +903,24 @@ int main(int argc, char *argv[]) } if (pFile) { +// char *ptr; + char text[BTUI_MAX_STRING_LENGTH_PER_LINE]; - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, - pFile); + if (!fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) + fprintf(stderr, "fail to fgets"); + sscanf(text, "%02x%02x", &bdaddr[0], &bdaddr[1]); - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, - pFile); + if (!fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) + fprintf(stderr, "fail to fgets"); + sscanf(text, "%02x", &bdaddr[2]); - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, - pFile); + if (!fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) + fprintf(stderr, "fail to fgets"); + sscanf(text, "%02x%02x%02x", &bdaddr[3], &bdaddr[4], &bdaddr[5]); diff --git a/tools/bcmtool_4358a1.c b/tools/bcmtool_4358a1.c index b35da8e..8f6d95e 100644 --- a/tools/bcmtool_4358a1.c +++ b/tools/bcmtool_4358a1.c @@ -238,7 +238,9 @@ UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 *p_param_buf) dump(pbuf, param_len+4); - write(fd, pbuf, param_len+4); + if (write(fd, pbuf, param_len+4) < 0) + DEBUG0("Fail to write pbuf"); + return 0; } @@ -346,6 +348,7 @@ UINT8 DownloadPatchram( char *patchram1 ) UINT32 len; char prm[128] ={0,}; FILE* pFile = NULL; + size_t size; INT32 FileSize=0; INT32 SentSize=0; @@ -385,9 +388,11 @@ UINT8 DownloadPatchram( char *patchram1 ) len = buffer[3]; - fread(&buffer[4],sizeof(UINT8),len, pFile); + size = fread(&buffer[4], sizeof(UINT8), len, pFile); + fprintf(stderr, "fread size: %d\n", size); - write(fd, buffer, len + 4); + size = write(fd, buffer, len + 4); + fprintf(stderr, "write size: %d\n", size); /* dispaly progress*/ SentSize += (len + 3); @@ -781,13 +786,16 @@ int main(int argc, char *argv[]) { char text[BTUI_MAX_STRING_LENGTH_PER_LINE]; - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + if (!fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) + fprintf(stderr, "fail to fgets"); sscanf(text,"%02x%02x",&bdaddr[0],&bdaddr[1]); - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + if (!fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) + fprintf(stderr, "fail to fgets"); sscanf(text,"%02x",&bdaddr[2]); - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile); + if (!fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, pFile)) + fprintf(stderr, "fail to fgets"); sscanf(text,"%02x%02x%02x",&bdaddr[3],&bdaddr[4],&bdaddr[5]); fprintf(stderr,"Writing B/D Address = %02X:%02X:%02X:%02X:%02X:%02X\n",bdaddr[0],bdaddr[1],bdaddr[2],bdaddr[3],bdaddr[4],bdaddr[5]); -- 2.7.4 From 68d777f327e1d9e7af1e7bf113039bc2818ddc02 Mon Sep 17 00:00:00 2001 From: DoHyun Pyun Date: Mon, 23 May 2016 10:45:07 +0900 Subject: [PATCH 10/16] Fix build error for common profile Change-Id: I8a7ae883ee8f4863712e5ef3b3f56abebd5dcd3e Signed-off-by: DoHyun Pyun --- tools/bcmtool_4330b1.c | 8 ++++---- tools/bcmtool_4343w.c | 4 ++-- tools/bcmtool_4358a1.c | 8 ++++---- 3 files changed, 10 insertions(+), 10 deletions(-) diff --git a/tools/bcmtool_4330b1.c b/tools/bcmtool_4330b1.c index e9947f6..f8cbb7e 100644 --- a/tools/bcmtool_4330b1.c +++ b/tools/bcmtool_4330b1.c @@ -388,11 +388,11 @@ UINT8 DownloadPatchram( char *patchram1 ) len = buffer[3]; - size = fread(&buffer[4], sizeof(UINT8), len, pFile); - fprintf(stderr, "fread size: %d\n", size); + size = fread(&buffer[4], sizeof(UINT8), len, pFile); + fprintf(stderr, "fread size: %d\n", (int) size); - size = write(fd, buffer, len + 4); - fprintf(stderr, "write size: %d\n", size); + size = write(fd, buffer, len + 4); + fprintf(stderr, "write size: %d\n", (int) size); /* dispaly progress*/ SentSize += (len + 3); diff --git a/tools/bcmtool_4343w.c b/tools/bcmtool_4343w.c index 4fdbfc4..d75cecf 100644 --- a/tools/bcmtool_4343w.c +++ b/tools/bcmtool_4343w.c @@ -392,10 +392,10 @@ UINT8 DownloadPatchram(char *patchram1) len = buffer[3]; size = fread(&buffer[4], sizeof(UINT8), len, pFile); - fprintf(stderr, "fread size: %d\n", size); + fprintf(stderr, "fread size: %d\n", (int) size); size = write(fd, buffer, len + 4); - fprintf(stderr, "write size: %d\n", size); + fprintf(stderr, "write size: %d\n", (int) size); /* dispaly progress */ SentSize += (len + 3); diff --git a/tools/bcmtool_4358a1.c b/tools/bcmtool_4358a1.c index 8f6d95e..c4e3b35 100644 --- a/tools/bcmtool_4358a1.c +++ b/tools/bcmtool_4358a1.c @@ -388,11 +388,11 @@ UINT8 DownloadPatchram( char *patchram1 ) len = buffer[3]; - size = fread(&buffer[4], sizeof(UINT8), len, pFile); - fprintf(stderr, "fread size: %d\n", size); + size = fread(&buffer[4], sizeof(UINT8), len, pFile); + fprintf(stderr, "fread size: %d\n", (int) size); - size = write(fd, buffer, len + 4); - fprintf(stderr, "write size: %d\n", size); + size = write(fd, buffer, len + 4); + fprintf(stderr, "write size: %d\n", (int) size); /* dispaly progress*/ SentSize += (len + 3); -- 2.7.4 From 3e4862098213d49c310c172f29a6618998752c40 Mon Sep 17 00:00:00 2001 From: Pawel Wieczorek Date: Thu, 19 May 2016 16:40:31 +0200 Subject: [PATCH 11/16] Provide virtual package bluetooth-scripts This patch allows "bluetooth-firmware-bcm*" packages to be used when "bluetooth-scripts" package is required as a dependency. Virtual package "bluetooth-scripts" was introduced to resolve file conflict between this project and "bluetooth-tools". Change-Id: I3636d4995e271e63618791b304241feb9d8e2d32 Signed-off-by: Pawel Wieczorek --- packaging/bluetooth-firmware-bcm.spec | 3 +++ 1 file changed, 3 insertions(+) diff --git a/packaging/bluetooth-firmware-bcm.spec b/packaging/bluetooth-firmware-bcm.spec index 0a35398..e332944 100644 --- a/packaging/bluetooth-firmware-bcm.spec +++ b/packaging/bluetooth-firmware-bcm.spec @@ -7,6 +7,7 @@ License: Apache # NOTE: Source name does not match package name. This should be # resolved in the future, by I don't have that power. - Ryan Ware Source0: %{name}-%{version}.tar.gz +Provides: bluetooth-scripts BuildRequires: pkgconfig(vconf) BuildRequires: cmake @@ -17,6 +18,7 @@ BuildRequires: cmake %package exynos3250 Summary: bcm firmware and tools for exynos3250 Group: Hardware Support/Handset +Provides: bluetooth-scripts %description exynos3250 bcm firmware and tools for exynos3250 @@ -24,6 +26,7 @@ bcm firmware and tools for exynos3250 %package artik Summary: bcm firmware and tools for artik Group: Hardware Support/Handset +Provides: bluetooth-scripts %description artik bcm firmware and tools for artik -- 2.7.4 From 5066d54cf230db1915210fa25521d29abf545a00 Mon Sep 17 00:00:00 2001 From: Taejin Woo Date: Thu, 23 Jun 2016 10:47:27 +0900 Subject: [PATCH 12/16] Fix script's execution path Change-Id: I4a43ef7a19bce5709f060c22e3937d52fdd36781 Signed-off-by: Taejin Woo --- packaging/bluetooth-firmware-bcm.spec | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/packaging/bluetooth-firmware-bcm.spec b/packaging/bluetooth-firmware-bcm.spec index e332944..64106c5 100644 --- a/packaging/bluetooth-firmware-bcm.spec +++ b/packaging/bluetooth-firmware-bcm.spec @@ -59,6 +59,10 @@ cat %{_builddir}/%{name}-%{version}/LICENSE.Broadcom >> %{buildroot}/usr/share/l %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-start.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-set-addr.sh +%post exynos3250 +rm -rf %{_prefix}/etc/bluetooth/bt-dev-start.sh +ln -s %{_prefix}/etc/bluetooth/bt-dev-start-exynos3250.sh %{_prefix}/etc/bluetooth/bt-dev-start.sh + %files exynos3250 %defattr(-,root,root,-) %{_bindir}/bcmtool_4343w -- 2.7.4 From 465aa02962303cc3949b0ea12d50420132b2e786 Mon Sep 17 00:00:00 2001 From: Taejin Woo Date: Thu, 23 Jun 2016 11:41:47 +0900 Subject: [PATCH 13/16] Add manifast file for rule Change-Id: I268ae5fc71690ffbe664378376ef9c2da2c623a7 Signed-off-by: Taejin Woo --- bluetooth-firmware-bcm.manifest | 5 +++++ packaging/bluetooth-firmware-bcm.spec | 3 +++ 2 files changed, 8 insertions(+) create mode 100644 bluetooth-firmware-bcm.manifest diff --git a/bluetooth-firmware-bcm.manifest b/bluetooth-firmware-bcm.manifest new file mode 100644 index 0000000..75b0fa5 --- /dev/null +++ b/bluetooth-firmware-bcm.manifest @@ -0,0 +1,5 @@ + + + + + diff --git a/packaging/bluetooth-firmware-bcm.spec b/packaging/bluetooth-firmware-bcm.spec index 64106c5..b060c34 100644 --- a/packaging/bluetooth-firmware-bcm.spec +++ b/packaging/bluetooth-firmware-bcm.spec @@ -58,6 +58,7 @@ cat %{_builddir}/%{name}-%{version}/LICENSE.Broadcom >> %{buildroot}/usr/share/l %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-end.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-start.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-set-addr.sh +%manifest %{name}.manifest %post exynos3250 rm -rf %{_prefix}/etc/bluetooth/bt-dev-start.sh @@ -73,6 +74,7 @@ ln -s %{_prefix}/etc/bluetooth/bt-dev-start-exynos3250.sh %{_prefix}/etc/bluetoo %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-start-exynos3250.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-set-addr.sh /usr/share/license/%{name} +%manifest %{name}.manifest %post artik rm -rf %{_prefix}/etc/bluetooth/bt-dev-start.sh @@ -87,3 +89,4 @@ ln -s %{_prefix}/etc/bluetooth/bt-dev-start-artik.sh %{_prefix}/etc/bluetooth/bt %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-start-artik.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-set-addr.sh /usr/share/license/%{name} +%manifest %{name}.manifest -- 2.7.4 From cbc4a4078129a514fd58290a15bc0fe443f35a64 Mon Sep 17 00:00:00 2001 From: Taejin Woo Date: Thu, 23 Jun 2016 12:19:10 +0900 Subject: [PATCH 14/16] Add hci's command service file for Broadcom BT/FM request Change-Id: Ib958dda1bfd9bf1cdb89cb787793e86b0642c9f3 Signed-off-by: Taejin Woo --- packaging/bluetooth-firmware-bcm.spec | 11 +++++++++++ packaging/bluetooth-hci-device.service | 11 +++++++++++ packaging/bluetooth-hciattach@.service | 13 +++++++++++++ 3 files changed, 35 insertions(+) create mode 100644 packaging/bluetooth-hci-device.service create mode 100644 packaging/bluetooth-hciattach@.service diff --git a/packaging/bluetooth-firmware-bcm.spec b/packaging/bluetooth-firmware-bcm.spec index b060c34..457a544 100644 --- a/packaging/bluetooth-firmware-bcm.spec +++ b/packaging/bluetooth-firmware-bcm.spec @@ -7,6 +7,8 @@ License: Apache # NOTE: Source name does not match package name. This should be # resolved in the future, by I don't have that power. - Ryan Ware Source0: %{name}-%{version}.tar.gz +Source1: bluetooth-hciattach@.service +Source2: bluetooth-hci-device.service Provides: bluetooth-scripts BuildRequires: pkgconfig(vconf) @@ -47,6 +49,9 @@ mkdir -p %{buildroot}/usr/share/license cp LICENSE.APLv2 %{buildroot}/usr/share/license/%{name} cat %{_builddir}/%{name}-%{version}/LICENSE.Broadcom >> %{buildroot}/usr/share/license/%{name} +install -D -m 0644 %SOURCE1 %{buildroot}%{_libdir}/systemd/system/bluetooth-hciattach@.service +install -D -m 0644 %SOURCE2 %{buildroot}%{_libdir}/systemd/system/bluetooth-hci-device.service + %files %defattr(-,root,root,-) #%{_bindir}/bcmtool_4330b1 @@ -58,6 +63,8 @@ cat %{_builddir}/%{name}-%{version}/LICENSE.Broadcom >> %{buildroot}/usr/share/l %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-end.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-start.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-set-addr.sh +%exclude %{_libdir}/systemd/system/bluetooth-hciattach@.service +%exclude %{_libdir}/systemd/system/bluetooth-hci-device.service %manifest %{name}.manifest %post exynos3250 @@ -74,6 +81,8 @@ ln -s %{_prefix}/etc/bluetooth/bt-dev-start-exynos3250.sh %{_prefix}/etc/bluetoo %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-start-exynos3250.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-set-addr.sh /usr/share/license/%{name} +%{_libdir}/systemd/system/bluetooth-hciattach@.service +%{_libdir}/systemd/system/bluetooth-hci-device.service %manifest %{name}.manifest %post artik @@ -89,4 +98,6 @@ ln -s %{_prefix}/etc/bluetooth/bt-dev-start-artik.sh %{_prefix}/etc/bluetooth/bt %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-dev-start-artik.sh %attr(755,-,-) %{_prefix}/etc/bluetooth/bt-set-addr.sh /usr/share/license/%{name} +%{_libdir}/systemd/system/bluetooth-hciattach@.service +%{_libdir}/systemd/system/bluetooth-hci-device.service %manifest %{name}.manifest diff --git a/packaging/bluetooth-hci-device.service b/packaging/bluetooth-hci-device.service new file mode 100644 index 0000000..e05228c --- /dev/null +++ b/packaging/bluetooth-hci-device.service @@ -0,0 +1,11 @@ +[Unit] +Description=hciattach service for Broadcom BT/FM requested +StopWhenUnneeded=yes + +[Service] +Type=oneshot +ExecStart=/usr/etc/bluetooth/bt-dev-start.sh +ExecStop=/usr/etc/bluetooth/bt-dev-end.sh +RemainAfterExit=yes +StandardOutput=journal+console +StandardError=inherit diff --git a/packaging/bluetooth-hciattach@.service b/packaging/bluetooth-hciattach@.service new file mode 100644 index 0000000..00ec9e9 --- /dev/null +++ b/packaging/bluetooth-hciattach@.service @@ -0,0 +1,13 @@ +[Unit] +Description=hciattach service requested by %I +Requires=bluetooth-hci-device.service +After=bluetooth-hci-device.service + +[Service] +Type=oneshot +ExecStartPre=-/bin/mkdir -p /run/bluetooth/ +ExecStart=/bin/touch /run/bluetooth/%I +ExecStop=/bin/rm /run/bluetooth/%I +RemainAfterExit=yes +StandardOutput=journal+console +StandardError=inherit -- 2.7.4 From 150a2def1d8bd8036856e9dad9c582b48b9737cf Mon Sep 17 00:00:00 2001 From: Hyuk Lee Date: Fri, 24 Jun 2016 14:01:30 +0900 Subject: [PATCH 15/16] Modify the license version Change-Id: Idd6836467d225a986474d96f84a61e238604968b Signed-off-by: Hyuk Lee --- packaging/bluetooth-firmware-bcm.spec | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packaging/bluetooth-firmware-bcm.spec b/packaging/bluetooth-firmware-bcm.spec index 457a544..bcd5a33 100644 --- a/packaging/bluetooth-firmware-bcm.spec +++ b/packaging/bluetooth-firmware-bcm.spec @@ -3,7 +3,7 @@ Summary: firmware and tools for bluetooth Version: 0.1.3 Release: 1 Group: Hardware Support/Handset -License: Apache +License: Apache-2.0 # NOTE: Source name does not match package name. This should be # resolved in the future, by I don't have that power. - Ryan Ware Source0: %{name}-%{version}.tar.gz -- 2.7.4 From 30cb3d4ceef424e31d6622c2b4d0666dc57cb061 Mon Sep 17 00:00:00 2001 From: DoHyun Pyun Date: Mon, 27 Jun 2016 16:42:19 +0900 Subject: [PATCH 16/16] Remove unused file Change-Id: I90b67d34974fbab6cbb52fdb1ad8a217708a93e3 Signed-off-by: DoHyun Pyun --- tools/bcmtool_4330b1_m.c | 1034 ---------------------------------------------- 1 file changed, 1034 deletions(-) delete mode 100644 tools/bcmtool_4330b1_m.c diff --git a/tools/bcmtool_4330b1_m.c b/tools/bcmtool_4330b1_m.c deleted file mode 100644 index 4934020..0000000 --- a/tools/bcmtool_4330b1_m.c +++ /dev/null @@ -1,1034 +0,0 @@ -/* - * Name: bcmtool_4330b1.c - * - * Description: Download a patchram files for the HCD format - * - * Copyright (c) 2012-2013, Broadcom Corp., All Rights Reserved. - * - * 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. - * - */ - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#define N_HCI 15 -#define HCI_UART_H4 0 -#define HCI_UART_BCSP 1 -#define HCI_UART_3WIRE 2 -#define HCI_UART_H4DS 3 -#define HCI_UART_LL 4 -#define HCIUARTSETPROTO _IOW('U', 200, int) -#define HCIUARTGETPROTO _IOR('U', 201, int) -#define HCIUARTGETDEVICE _IOR('U', 202, int) - -/* Pre baudrate change for fast download */ -#define HIGH_SPEED_PATCHRAM_DOWNLOAD TRUE - -/* Host Stack Idle Threshold */ -#define HCILP_IDLE_THRESHOLD 0x01 - -/* Host Controller Idle Threshold */ -#define HCILP_HC_IDLE_THRESHOLD 0x01 - -/* BT_WAKE Polarity - 0=Active Low, 1= Active High */ -#define HCILP_BT_WAKE_POLARITY 1 - -/* HOST_WAKE Polarity - 0=Active Low, 1= Active High */ -#define HCILP_HOST_WAKE_POLARITY 1 - -/* Local Feature */ -#define BCM_DISABLE_RF_PWRCTRL FALSE - -#define RELEASE_DATE "2011.02.07" -#define DEBUG 1 - -/* Broadcom AXI patch for BCM4335 chipset only */ -#define DEPLOY_4335A_AXI_BRIDGE_PATCH TRUE - -/* The fix for AXI bridge contention between BT and WLAN: - * - * Set this TRUE only when - * 1. the platform is using BCM4335A or BCM4335B0, and - * 2. Kernel source has implemented AXI BRIDGE lock logic. - */ -#ifndef DEPLOY_4335A_AXI_BRIDGE_PATCH -#define DEPLOY_4335A_AXI_BRIDGE_PATCH FALSE -#endif - -typedef unsigned char UINT8; -typedef unsigned short UINT16; -typedef unsigned long UINT32; -typedef signed long INT32; -typedef signed char INT8; -typedef signed short INT16; -typedef unsigned char BOOLEAN; - -#define FALSE 0 -#define TRUE (!FALSE) - -#define BD_ADDR_LEN 6 /* Device address length */ -typedef UINT8 BD_ADDR[BD_ADDR_LEN]; /* Device address */ - -#define HCI_GRP_LINK_CONTROL_CMDS (0x01 << 10) -#define HCI_GRP_LINK_POLICY_CMDS (0x02 << 10) -#define HCI_GRP_HOST_CONT_BASEBAND_CMDS (0x03 << 10) -#define HCI_GRP_INFORMATIONAL_PARAMS (0x04 << 10) -#define HCI_GRP_STATUS_PARAMS (0x05 << 10) -#define HCI_GRP_TESTING_CMDS (0x06 << 10) -#define HCI_GRP_L2CAP_CMDS (0x07 << 10) -#define HCI_GRP_L2CAP_HCI_EVTS (0x08 << 10) -#define HCI_GRP_VENDOR_SPECIFIC (0x3F << 10) - -#define HCI_RESET (0x0003 | HCI_GRP_HOST_CONT_BASEBAND_CMDS) -#define HCI_SET_EVENT_FILTER (0x0005 | HCI_GRP_HOST_CONT_BASEBAND_CMDS) -#define HCI_READ_LOCAL_NAME (0x0014 | HCI_GRP_HOST_CONT_BASEBAND_CMDS) -#define HCI_READ_SCAN_ENABLE (0x0019 | HCI_GRP_HOST_CONT_BASEBAND_CMDS) -#define HCI_WRITE_SCAN_ENABLE (0x001A | HCI_GRP_HOST_CONT_BASEBAND_CMDS) - -#define HCI_READ_LOCAL_VERSION_INFO (0x0001 | HCI_GRP_INFORMATIONAL_PARAMS) -#define HCI_READ_LOCAL_FEATURES (0x0003 | HCI_GRP_INFORMATIONAL_PARAMS) - -#define HCI_ENABLE_DEV_UNDER_TEST_MODE (0x0003 | HCI_GRP_TESTING_CMDS) - -#define HCI_BRCM_SUPER_PEEK_POKE (0x000A | HCI_GRP_VENDOR_SPECIFIC) -#define VSC_WRITE_BD_ADDR (0x0001 | HCI_GRP_VENDOR_SPECIFIC) -#define VSC_HCI_CMD_SET_LOC_FEATURES_CMD (0x000B | HCI_GRP_VENDOR_SPECIFIC) -#define HCI_BRCM_UPDATE_BAUDRATE_CMD (0x0018 | HCI_GRP_VENDOR_SPECIFIC) -#define HCI_BRCM_WRITE_SCO_PCM_INT_PARAM (0x001C | HCI_GRP_VENDOR_SPECIFIC) -#define VSC_WRITE_PCM_DATA_FORMAT_PARAM (0x001E | HCI_GRP_VENDOR_SPECIFIC) -#define HCI_BRCM_WRITE_SLEEP_MODE (0x0027 | HCI_GRP_VENDOR_SPECIFIC) -#define HCI_BRCM_DOWNLOAD_MINI_DRV (0x002E | HCI_GRP_VENDOR_SPECIFIC) -#define VSC_WRITE_UART_CLOCK_SETTING (0x0045 | HCI_GRP_VENDOR_SPECIFIC) -#define HCI_VSC_WRITE_RAM (0x004C | HCI_GRP_VENDOR_SPECIFIC) -#define HCI_VSC_LAUNCH_RAM (0x004E | HCI_GRP_VENDOR_SPECIFIC) - -#define VOICE_SETTING_MU_LAW_MD 0x0100 -#define VOICE_SETTING_LINEAR_MD 0x0060 - -#define HCI_ARM_MEM_PEEK 0x04 -#define HCI_ARM_MEM_POKE 0x05 - -#define BTUI_MAX_STRING_LENGTH_PER_LINE 255 -#define HCI_BRCM_WRITE_SLEEP_MODE_LENGTH 10 - -#define HCI_BRCM_UPDATE_BAUD_RATE_ENCODED_LENGTH 0x02 -#define HCI_BRCM_UPDATE_BAUD_RATE_UNENCODED_LENGTH 0x06 - -#define VSC_WRITE_UART_CLOCK_SETTING_LEN 1 - -/* print string with time stamp */ -#define TDEBUG0(m) if(debug_mode) {print_time();fprintf(stderr,m);} -#define TDEBUG1(m,n1) if(debug_mode) {print_time();fprintf(stderr,m,n1);} -#define TDEBUG2(m,n1,n2) if(debug_mode) {print_time();fprintf(stderr,m,n1,n2);} -#define TDEBUG3(m,n1,n2,n3) if(debug_mode) {print_time();fprintf(stderr,m,n1,n2,n3);} -#define TDEBUG4(m,n1,n2,n3,n4) if(debug_mode) {print_time();fprintf(stderr,m,n1,n2,n3,n4);} -#define TDEBUG5(m,n1,n2,n3,n4,n5) if(debug_mode) {print_time();fprintf(stderr,m,n1,n2,n3,n4,n5);} -#define TDEBUG6(m,n1,n2,n3,n4,n5,n6) if(debug_mode) {print_time();fprintf(stderr,m,n1,n2,n3,n4,n5,n6);} - -/* print just string */ -#define DEBUG0(m) if(debug_mode) {fprintf(stderr,m);} -#define DEBUG1(m,n1) if(debug_mode) {fprintf(stderr,m,n1);} -#define DEBUG2(m,n1,n2) if(debug_mode) {fprintf(stderr,m,n1,n2);} -#define DEBUG3(m,n1,n2,n3) if(debug_mode) {fprintf(stderr,m,n1,n2,n3);} -#define DEBUG4(m,n1,n2,n3,n4) if(debug_mode) {fprintf(stderr,m,n1,n2,n3,n4);} -#define DEBUG5(m,n1,n2,n3,n4,n5) if(debug_mode) {fprintf(stderr,m,n1,n2,n3,n4,n5);} -#define DEBUG6(m,n1,n2,n3,n4,n5,n6) if(debug_mode) {fprintf(stderr,m,n1,n2,n3,n4,n5,n6);} - -#define STREAM_TO_UINT8(u8, p) {u8 = (UINT8)(*(p)); (p) += 1;} -#define STREAM_TO_UINT16(u16, p) {u16 = ((UINT16)(*(p)) + (((UINT16)(*((p) + 1))) << 8)); (p) += 2;} -#define STREAM_TO_UINT32(u32, p) {u32 = (((UINT32)(*(p))) + ((((UINT32)(*((p) + 1)))) << 8) + ((((UINT32)(*((p) + 2)))) << 16) + ((((UINT32)(*((p) + 3)))) << 24)); (p) += 4;} - -#define ROTATE_BD_ADDR(p1, p2) \ - do \ - { \ - p1[0] = p2[5]; \ - p1[1] = p2[4]; \ - p1[2] = p2[3]; \ - p1[3] = p2[2]; \ - p1[4] = p2[1]; \ - p1[5] = p2[0]; \ - } while (0) - -UINT8 vsc_for_pcm_config[5] = { 0x00, 0x00, 0x03, 0x03, 0x00 }; - -/* - Byte1 -- 0 for MSb first - Byte2 -- 0 Fill value - Byte3 -- 1 Fill option (0:0's, 1:1's , 2:Signed, 3:Programmable) - Byte4 -- 1 Number of fill bits - Byte5 -- 1 Right justified (0 for left justified) -*/ - -UINT8 vsc_for_sco_pcm[5] = { 0x00, 0x01, 0x00, 0x01, 0x01 }; - -/* - Neverland : PCM, 256, short, master ,master - Volance : PCM, 256, short, master ,master - - Byte1 -- 0 for PCM 1 for UART or USB - Byte2 -- 0 : 128, 1: 256, 2:512, 3:1024, 4:2048 Khz - Byte3 -- 0 for short frame sync 1 for long frame sync - Byte4 -- 0 Clock direction 0 for same as sync 1 for opposite direction - Byte5 -- 0 for slave 1 for master -*/ - -int fd; /* HCI handle */ - -BOOLEAN debug_mode = FALSE; /* Debug Mode Enable */ - -BOOLEAN use_two_stop_bits = FALSE; /* Flag of two stop bits for tty */ - -unsigned char buffer[1024]; - -struct termios termios; - -void ChangeBaudRate(UINT32 baudrate); - -void exit_err(UINT8 err) -{ -#if ( HIGH_SPEED_PATCHRAM_DOWNLOAD == TRUE ) - ChangeBaudRate(115200); -#endif - exit(err); -} - -void print_time(void) -{ -#if 0 - struct timespec tp; - int rs; - - rs = clock_gettime(CLOCK_REALTIME, &tp); - fprintf(stderr, "[%04d : %06d]\n", tp.tv_sec, tp.tv_nsec / 1000); - return; -#endif -} - -void dump(unsigned char *out, int len) -{ - int i; - - for (i = 0; i < len; i++) { - if (!(i % 16)) { - DEBUG0("\n"); - } - DEBUG1("%02x ", out[i]); - } - DEBUG0("\n\n"); -} - -UINT8 SendCommand(UINT16 opcode, UINT8 param_len, UINT8 * p_param_buf) -{ - UINT8 pbuf[255] = { 0, }; - UINT8 i = 0; - - pbuf[0] = 0x1; - pbuf[1] = (UINT8) (opcode); - pbuf[2] = (UINT8) (opcode >> 8); - pbuf[3] = param_len; - - for (i = 0; i < param_len; i++) { - pbuf[i + 4] = *p_param_buf++; - } - - DEBUG1("Send %d", param_len + 4); - - dump(pbuf, param_len + 4); - - write(fd, pbuf, param_len + 4); - return 0; -} - -void expired(int sig) -{ - static UINT8 count = 0; - DEBUG0("expired try again\n"); - SendCommand(HCI_RESET, 0, NULL); - alarm(1); - count++; - - if (count > 3) { - fprintf(stderr, "[ERR] HCI reset time expired\n"); - exit(1); - } -} - -void read_event(int fd, unsigned char *buffer) -{ - int i = 0; - int len = 3; - int count; - - while ((count = read(fd, &buffer[i], len)) < len) { - i += count; - len -= count; - } - - i += count; - len = buffer[2]; - - while ((count = read(fd, &buffer[i], len)) < len) { - i += count; - len -= count; - } - -#ifdef DEBUG - count += i; - - DEBUG1("\nreceived %d", count); - dump(buffer, count); -#endif -} - -INT32 filesize(char *name) -{ - INT32 size; - int flag; - struct stat buf; - - flag = stat(name, &buf); - if (flag == -1) - return -1; - - size = buf.st_size; - return (size); -} - -void DisplayProgress(int total, int val) -{ -#if 0 -#define PROGRESS_NUM 20 - - int p; - int i; - char text[PROGRESS_NUM + 2] = { 0, }; - - text[0] = '['; - text[PROGRESS_NUM + 1] = ']'; - p = (val * PROGRESS_NUM) / total; - - for (i = 1; i <= p; i++) { - text[i] = '='; - } - - for (i = p + 1; i <= PROGRESS_NUM; i++) { - text[i] = ' '; - } - - for (i = 0; i <= (PROGRESS_NUM + 1); i++) { - fprintf(stderr, "%c", text[i]); - } - - if (p >= PROGRESS_NUM) - fprintf(stderr, " %6d/%6d\n", val, total); - else - fprintf(stderr, " %6d/%6d\r", val, total); -#else - if (val == total) - fprintf(stderr, " %6d/%6d\n", val, total); - else - fprintf(stderr, " %6d/%6d\r", val, total); -#endif -} - -#if (DEPLOY_4335A_AXI_BRIDGE_PATCH == TRUE) -/* 4335A/B0 AXI BRIDEG lock cookie */ -struct btlock { - int lock; - int cookie; -}; - -#define AXI_LOCK_COOKIE ('B' | 'T'<<8 | '3'<<16 | '5'<<24) /* BT35 */ -#define AXI_LOCK_FS_NODE "/dev/btlock" - -static const UINT8 bcm4335a_axi_patch_addr[4] = -{ - 0x00, 0x02, 0x0d, 0x00 -}; - -static const UINT8 bcm4335a_axi_patch[] = -{ - 0x00, 0x02, 0x0d, 0x00, /* bcm4335a_axi_patch_addr */ - 0x70, 0xb5, 0x0c, 0x49, 0x4c, 0xf6, 0x20, 0x30, - 0x8a, 0xf7, 0xbb, 0xf9, 0x01, 0x28, 0x0d, 0xd1, - 0x8a, 0xf7, 0x80, 0xf9, 0x08, 0xb1, 0x04, 0x25, - 0x00, 0xe0, 0x05, 0x25, 0x00, 0x24, 0x03, 0xe0, - 0x8a, 0xf7, 0x74, 0xf9, 0x64, 0x1c, 0xe4, 0xb2, - 0xac, 0x42, 0xf9, 0xd3, 0xbd, 0xe8, 0x70, 0x40, - 0x8a, 0xf7, 0x7f, 0xb9, 0xb0, 0x9b, 0x04, 0x00 -}; - -/******************************************************************************* -** -** Function hw_4335_dl_axi_patch -** -** Description Download 4335Ax/4335B0 AXI BRIDGE patch -** -** Returns TRUE, if fw patch is sent -** FALSE, otherwise -** -*******************************************************************************/ -static UINT8 hw_4335_dl_axi_patch(void) -{ - SendCommand(HCI_VSC_WRITE_RAM, sizeof(bcm4335a_axi_patch), - (UINT8 *) bcm4335a_axi_patch); - read_event(fd, buffer); - - DEBUG0("hw_4335_dl_axi_patch downloading done"); - - SendCommand(HCI_VSC_LAUNCH_RAM, sizeof(bcm4335a_axi_patch_addr), - (UINT8 *) bcm4335a_axi_patch_addr); - read_event(fd, buffer); - - DEBUG0("hw_4335_dl_axi_patch launching done"); - - return TRUE; -} - -/******************************************************************************* -** -** Function hw_4335_release_axi_bridge_lock -** -** Description Notify kernel to release the AXI BRIDGE lock which was -** acquired earlier in rfkill driver when powering on BT -** Controller -** -** Returns None -** -*******************************************************************************/ -void hw_4335_axi_bridge_lock(void) -{ - int fd, ret; - struct btlock lock; - - lock.cookie = AXI_LOCK_COOKIE; - lock.lock = 1; - - fd = open(AXI_LOCK_FS_NODE, O_RDWR); - if (fd >= 0) - { - ret = write(fd, &lock, sizeof(lock)); - DEBUG0("4335 AXI BRIDGE lock"); - close(fd); - } - else - { - DEBUG1("Failed to unlock AXI LOCK -- can't open %s", AXI_LOCK_FS_NODE); - } -} - -/******************************************************************************* -** -** Function hw_4335_release_axi_bridge_lock -** -** Description Notify kernel to release the AXI BRIDGE lock which was -** acquired earlier in rfkill driver when powering on BT -** Controller -** -** Returns None -** -*******************************************************************************/ -void hw_4335_release_axi_bridge_lock(void) -{ - int fd, ret; - struct btlock lock; - - lock.cookie = AXI_LOCK_COOKIE; - lock.lock = 0; - - fd = open(AXI_LOCK_FS_NODE, O_RDWR); - if (fd >= 0) - { - ret = write(fd, &lock, sizeof(lock)); - DEBUG0("Releasing 4335 AXI BRIDGE lock"); - close(fd); - } - else - { - DEBUG1("Failed to unlock AXI LOCK -- can't open %s", AXI_LOCK_FS_NODE); - } -} -#endif /* (DEPLOY_4335A_AXI_BRIDGE_PATCH == TRUE) */ - -UINT8 DownloadPatchram(char *patchram1) -{ - UINT32 len; - char prm[128] = { 0, }; - FILE *pFile = NULL; - - INT32 FileSize = 0; - INT32 SentSize = 0; - - DEBUG1("\n%s\n", patchram1); - - /* HCI reset */ - DEBUG0("HCI reset\n"); - SendCommand(HCI_RESET, 0, NULL); - alarm(1); - read_event(fd, buffer); - alarm(0); - -#if (DEPLOY_4335A_AXI_BRIDGE_PATCH == TRUE) - char *p_tmp; - - SendCommand(HCI_READ_LOCAL_NAME, 0, NULL); - read_event(fd, buffer); - - p_tmp = strstr((char *)(buffer + 7), "BCM4335"); - DEBUG1( "chip_name [%s]\n", p_tmp); - - if ((p_tmp != NULL) && - ((p_tmp[7] == 'A') /* 4335A */|| - ((p_tmp[7] == 'B') && (p_tmp[8] == '0')) /* 4335B0 */)) { - hw_4335_dl_axi_patch(); - } - hw_4335_release_axi_bridge_lock(); -#endif - -#if ( HIGH_SPEED_PATCHRAM_DOWNLOAD == TRUE ) - ChangeBaudRate(921600); -#endif - - strncpy(prm, patchram1, 127); - - fprintf(stderr, "Download Start\n"); - - if ((pFile = fopen(prm, "r")) == NULL) { - fprintf(stderr, "file %s could not be opened, error %d\n", prm, - errno); - exit_err(1); - } - FileSize = filesize(prm); - - SendCommand(HCI_BRCM_DOWNLOAD_MINI_DRV, 0, NULL); - read_event(fd, buffer); - - usleep(50000); - - while (fread(&buffer[1], sizeof(UINT8), 3, pFile)) { - buffer[0] = 0x01; - - len = buffer[3]; - - fread(&buffer[4], sizeof(UINT8), len, pFile); - - write(fd, buffer, len + 4); - - /* dispaly progress */ - SentSize += (len + 3); -#if 0 - DisplayProgress(FileSize, SentSize); -#endif - /* dispaly progress */ - - read_event(fd, buffer); - - } - fclose(pFile); - - usleep(100000); /*100ms delay */ - - tcflush(fd, TCIOFLUSH); - tcgetattr(fd, &termios); - cfmakeraw(&termios); - termios.c_cflag |= CRTSCTS; - - if (use_two_stop_bits) - termios.c_cflag |= CSTOPB; - - tcsetattr(fd, TCSANOW, &termios); - tcflush(fd, TCIOFLUSH); - tcsetattr(fd, TCSANOW, &termios); - tcflush(fd, TCIOFLUSH); - tcflush(fd, TCIOFLUSH); - cfsetospeed(&termios, B115200); - cfsetispeed(&termios, B115200); - tcsetattr(fd, TCSANOW, &termios); - - /* Send HCI_RESET Command and process event */ - DEBUG0("HCI reset\n"); - SendCommand(HCI_RESET, 0, NULL); - alarm(1); - read_event(fd, buffer); - alarm(0); - fprintf(stderr, "Download Complete\n"); - - return 0; -} - -void SetScanEnable(void) -{ - UINT8 scan_data[1]; - - /* 0x00: No scan enabled */ - /* 0x01: Inquiry scan enabled | Page scan disabled */ - /* 0x02: Inquiry scan disabled | Page scan enabled */ - /* 0x03: Inquiry scan enabled | Page scan enabled */ - - scan_data[0] = 0x03; - SendCommand(HCI_WRITE_SCAN_ENABLE, 1, &scan_data[0]); - read_event(fd, buffer); -} - -void SetAudio(void) -{ - fprintf(stderr, "Write Audio parameter\n"); - - DEBUG5("vsc_for_sco_pcm = {%d,%d,%d,%d,%d}\n", vsc_for_sco_pcm[0], - vsc_for_sco_pcm[1], vsc_for_sco_pcm[2], - vsc_for_sco_pcm[3], vsc_for_sco_pcm[4]); - - SendCommand(HCI_BRCM_WRITE_SCO_PCM_INT_PARAM, 5, - (UINT8 *) vsc_for_sco_pcm); - read_event(fd, buffer); - - DEBUG5("vsc_for_pcm_config = {%d,%d,%d,%d,%d}\n", vsc_for_pcm_config[0], - vsc_for_pcm_config[1], vsc_for_pcm_config[2], - vsc_for_pcm_config[3], vsc_for_pcm_config[4]); - - SendCommand(VSC_WRITE_PCM_DATA_FORMAT_PARAM, 5, - (UINT8 *) vsc_for_pcm_config); - read_event(fd, buffer); -} - -void SetPcmConf(UINT8 p0, UINT8 p1, UINT8 p2, UINT8 p3, UINT8 p4) -{ - vsc_for_pcm_config[0] = p0; - vsc_for_pcm_config[1] = p1; - vsc_for_pcm_config[2] = p2; - vsc_for_pcm_config[3] = p3; - vsc_for_pcm_config[4] = p4; -} - -void SetScoConf(UINT8 p0, UINT8 p1, UINT8 p2, UINT8 p3, UINT8 p4) -{ - vsc_for_sco_pcm[0] = p0; - vsc_for_sco_pcm[1] = p1; - vsc_for_sco_pcm[2] = p2; - vsc_for_sco_pcm[3] = p3; - vsc_for_sco_pcm[4] = p4; -} - -void HCILP_Enable(BOOLEAN on) -{ - fprintf(stderr, "Set Low Power mode %d\n", on); - UINT8 data[HCI_BRCM_WRITE_SLEEP_MODE_LENGTH] = { - 0x01, /* Sleep Mode algorithm 1 */ - HCILP_IDLE_THRESHOLD, /* Host Idle Treshold in 300ms */ - HCILP_HC_IDLE_THRESHOLD, /* Host Controller Idle Treshold in 300ms *//* this should be less than scan interval. */ - HCILP_BT_WAKE_POLARITY, /* BT_WAKE Polarity - 0=Active Low, 1= Active High */ - HCILP_HOST_WAKE_POLARITY, /* HOST_WAKE Polarity - 0=Active Low, 1= Active High */ - 0x01, /* Allow host Sleep during SCO */ - 0x01, /* Combine Sleep Mode and LPM - The device will not sleep in mode 0 if this flag is set to 1, */ - 0x00, /* UART_TXD Tri-State : 0x00 = Do not tri-state UART_TXD in sleep mode */ - 0x00, /* NA to Mode 1 */ - 0x00, /* NA to Mode 1 */ - }; - - if (on) { - data[0] = 0x01; - } else { - data[0] = 0x00; - } - - SendCommand(HCI_BRCM_WRITE_SLEEP_MODE, HCI_BRCM_WRITE_SLEEP_MODE_LENGTH, - (UINT8 *) data); - read_event(fd, buffer); -} - -UINT32 uart_speed(UINT32 Speed) -{ - switch (Speed) { - case 115200: - return B115200; - case 230400: - return B230400; - case 460800: - return B460800; - case 921600: - return B921600; - case 1000000: - return B1000000; - case 1500000: - return B1500000; - case 2000000: - return B2000000; - case 2500000: - return B2500000; - case 3000000: - return B3000000; - case 4000000: - return B4000000; - default: - return B115200; - } -} - -void ChangeBaudRate(UINT32 baudrate) -{ - UINT8 hci_data[HCI_BRCM_UPDATE_BAUD_RATE_UNENCODED_LENGTH] = - { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }; - UINT8 uart_clock_24 = 0x2; /* 0x1 - UART Clock 48MHz, 0x2 - UART Clock 24MHz */ - UINT8 uart_clock_48 = 0x1; /* 0x1 - UART Clock 48MHz, 0x2 - UART Clock 24MHz */ - - switch (baudrate) { - case 115200: - case 230400: - case 460800: - case 921600: - case 1000000: - case 1500000: - case 2000000: - case 2500000: - /* Write UART Clock setting of 24MHz */ - DEBUG0("Change UART_CLOCK 24Mhz\n"); - SendCommand(VSC_WRITE_UART_CLOCK_SETTING, - VSC_WRITE_UART_CLOCK_SETTING_LEN, - (UINT8 *) & uart_clock_24); - read_event(fd, buffer); - break; - - case 3000000: - case 4000000: - /* Write UART Clock setting of 48MHz */ - DEBUG0("Change UART_CLOCK 48Mh\nz"); - SendCommand(VSC_WRITE_UART_CLOCK_SETTING, - VSC_WRITE_UART_CLOCK_SETTING_LEN, - (UINT8 *) & uart_clock_48); - read_event(fd, buffer); - break; - - default: - fprintf(stderr, "Not Support baudrate = %ld\n", baudrate); - exit_err(1); - break; - } - - hci_data[2] = baudrate & 0xFF; - hci_data[3] = (baudrate >> 8) & 0xFF; - hci_data[4] = (baudrate >> 16) & 0xFF; - hci_data[5] = (baudrate >> 24) & 0xFF; - - DEBUG1("Change Baudrate %ld\n", baudrate); - - SendCommand(HCI_BRCM_UPDATE_BAUDRATE_CMD, - HCI_BRCM_UPDATE_BAUD_RATE_UNENCODED_LENGTH, - (UINT8 *) hci_data); - read_event(fd, buffer); - - tcflush(fd, TCIOFLUSH); - tcgetattr(fd, &termios); - cfmakeraw(&termios); - termios.c_cflag |= CRTSCTS; - - if (use_two_stop_bits) - termios.c_cflag |= CSTOPB; - - tcsetattr(fd, TCSANOW, &termios); - tcflush(fd, TCIOFLUSH); - tcsetattr(fd, TCSANOW, &termios); - tcflush(fd, TCIOFLUSH); - tcflush(fd, TCIOFLUSH); - cfsetospeed(&termios, uart_speed(baudrate)); - cfsetispeed(&termios, uart_speed(baudrate)); - tcsetattr(fd, TCSANOW, &termios); - -} - -void EnableTestMode(void) -{ - UINT8 filter_data[] = { 0x02, 0x00, 0x02 }; - - /* bt sleep disable */ - HCILP_Enable(FALSE); - - /* Enable both Inquiry & Page Scans */ - SetScanEnable(); - - /* Set Event Filter: Enable Auto Connect */ - SendCommand(HCI_SET_EVENT_FILTER, 0x03, (UINT8 *) filter_data); - read_event(fd, buffer); - - /* Enable Device under test */ - SendCommand(HCI_ENABLE_DEV_UNDER_TEST_MODE, 0x0, NULL); - read_event(fd, buffer); - - fprintf(stderr, "Enable Device Under Test\n"); -} - -void SetLocalFeatures(void) -{ - UINT8 *data = NULL; - - DEBUG0("Read Local Feature\n"); - SendCommand(HCI_READ_LOCAL_FEATURES, 0, NULL); - read_event(fd, buffer); - - data = &buffer[7]; - -#if (BCM_DISABLE_RF_PWRCTRL == TRUE) - fprintf(stderr, "Remove Power Control\n"); - data[2] &= 0xFB; /* Power contrel */ -#endif - DEBUG0("Write Local Feature\n"); - SendCommand(VSC_HCI_CMD_SET_LOC_FEATURES_CMD, 0x08, (UINT8 *) data); - read_event(fd, buffer); -} - -void EnbleHCI(void) -{ - int i = N_HCI; - int proto = HCI_UART_H4; - - if (ioctl(fd, TIOCSETD, &i) < 0) { - fprintf(stderr, "Can't set line discipline\n"); - return; - } - - if (ioctl(fd, HCIUARTSETPROTO, proto) < 0) { - fprintf(stderr, "Can't set hci protocol\n"); - return; - } - fprintf(stderr, "Done setting line discpline\n"); - return; - -} - -void print_usage(void) -{ - fprintf(stderr, "\n"); - fprintf(stderr, "BRCM BT tool for Linux release %s\n", RELEASE_DATE); - fprintf(stderr, "\n"); - fprintf(stderr, - " Usage: bcmtool [command parameter],....\n\n"); - fprintf(stderr, - " -FILE Patchram file name EX) -FILE=BCM43xx_xxx.hcd\n"); - fprintf(stderr, - " -BAUD Set Baudrate EX) -BAUD=3000000\n"); - fprintf(stderr, - " -ADDR BD addr file name EX) -ADDR=.bdaddr\n"); - fprintf(stderr, " -SCO Enable SCO/PCM config EX) -SCO\n"); - fprintf(stderr, - " -SETSCO SCO/PCM values verify EX) -SETSCO=0,1,0,1,1,0,0,3,3,0\n"); - fprintf(stderr, " -LP Enable Low power EX) -LP\n"); - fprintf(stderr, " -FEATURE Set local Feature EX) -FEATURE\n"); - fprintf(stderr, - " -DUT Enable DUT mode(do not use with -LP) EX) -DUT\n"); - fprintf(stderr, - " -ATTACH Attach BT controller to BlueZ stack EX) -ATTACH\n"); - fprintf(stderr, " -DEBUG Debug message EX) -DEBUG\n"); - fprintf(stderr, " -CSTOPB Set two stop bits for tty EX) -CSTOPB\n"); - fprintf(stderr, "\n"); -} - -int main(int argc, char *argv[]) -{ - UINT8 i = 0; - - if (argc < 2) { - print_usage(); - exit(1); - } else { - fprintf(stderr, "BRCM BT tool for Linux release %s\n", - RELEASE_DATE); - } - -#if (DEPLOY_4335A_AXI_BRIDGE_PATCH == TRUE) - hw_4335_axi_bridge_lock(); - usleep(50000); -#endif - - /* Open dev port */ - if ((fd = open(argv[1], O_RDWR | O_NOCTTY)) == -1) { - fprintf(stderr, "port %s could not be opened, error %d\n", - argv[1], errno); - exit(2); - } - - tcflush(fd, TCIOFLUSH); - tcgetattr(fd, &termios); - cfmakeraw(&termios); - termios.c_cflag |= CRTSCTS; - - if (use_two_stop_bits) - termios.c_cflag |= CSTOPB; - - tcsetattr(fd, TCSANOW, &termios); - tcflush(fd, TCIOFLUSH); - tcsetattr(fd, TCSANOW, &termios); - tcflush(fd, TCIOFLUSH); - tcflush(fd, TCIOFLUSH); - cfsetospeed(&termios, B115200); - cfsetispeed(&termios, B115200); - tcsetattr(fd, TCSANOW, &termios); - - signal(SIGALRM, expired); - - for (i = 2; i < argc; i++) { - char *ptr = argv[i]; - - if (strstr(ptr, "-DEBUG")) { - debug_mode = TRUE; - DEBUG0("DEBUG On\n"); - break; - } - - if (strstr(ptr, "-CSTOPB")) { - use_two_stop_bits = TRUE; - DEBUG0("Use two stop bits for tty\n"); - break; - } - } - - for (i = 2; i < argc; i++) { - char *ptr = argv[i]; - - if (ptr == NULL) - continue; - - fprintf(stderr, "[%d] %s\n", i - 1, ptr); - - if (strstr(ptr, "-FILE=")) { - char prm_name[128]; - - ptr += 6; - - strncpy(prm_name, ptr, 127); - DownloadPatchram(prm_name); - - } else if (strstr(ptr, "-BAUD=")) { - UINT32 baudrate; - - ptr += 6; - baudrate = atoi(ptr); - - ChangeBaudRate(baudrate); - } else if (strstr(ptr, "-ADDR=")) { - char *bdaddr_filename; - FILE *pFile = NULL; - - int bdaddr[10]; /* Displayed BD Address */ - - BD_ADDR local_addr; /* BD Address for write */ - -#if 0 - ptr += 6; - if (sscanf - (ptr, "%02X:%02X:%02X:%02X:%02X:%02X", &bdaddr[0], - &bdaddr[1], &bdaddr[2], &bdaddr[3], &bdaddr[4], - &bdaddr[5]) != 6) { - fprintf(stderr, "-ADDR: Parameter error"); - exit_err(1); - } - bte_write_bdaddr(bdaddr); -#endif - ptr += 6; - bdaddr_filename = ptr; - - if (bdaddr_filename) { - pFile = fopen(bdaddr_filename, "r"); - } - - if (pFile) { - char text[BTUI_MAX_STRING_LENGTH_PER_LINE]; - - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, - pFile); - sscanf(text, "%02x%02x", &bdaddr[0], - &bdaddr[1]); - - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, - pFile); - sscanf(text, "%02x", &bdaddr[2]); - - fgets(text, BTUI_MAX_STRING_LENGTH_PER_LINE, - pFile); - sscanf(text, "%02x%02x%02x", &bdaddr[3], - &bdaddr[4], &bdaddr[5]); - - fprintf(stderr, - "Writing B/D Address = %02X:%02X:%02X:%02X:%02X:%02X\n", - bdaddr[0], bdaddr[1], bdaddr[2], - bdaddr[3], bdaddr[4], bdaddr[5]); - - ROTATE_BD_ADDR(local_addr, bdaddr); - - SendCommand(VSC_WRITE_BD_ADDR, BD_ADDR_LEN, - (UINT8 *) local_addr); - read_event(fd, buffer); - } else { - fprintf(stderr, "-ADDR: file open fail\n"); - exit_err(1); - } - - } else if (strstr(ptr, "-SCO")) { - SetAudio(); - - } else if (strstr(ptr, "-SETSCO=")) { - ptr += 8; - int value[10]; - - if (sscanf - (ptr, "%d,%d,%d,%d,%d,%d,%d,%d,%d,%d", &value[0], - &value[1], &value[2], &value[3], &value[4], - &value[5], &value[6], &value[7], &value[8], - &value[9]) != 10) { - DEBUG0("PCM / SCO configuration value err\n"); - DEBUG0 - ("SCO_Routing,PCM_Interface_Rate,Frame_Type,Sync_Mode,Clock_Mode,LSB_First,Fill_bits,Fill_Method,Fill_Num,Right_Justify\n"); - exit_err(1); - } - - SetScoConf(value[0], value[1], value[2], value[3], - value[4]); - SetPcmConf(value[5], value[6], value[7], value[8], - value[9]); - SetAudio(); - } else if (strstr(ptr, "-LP")) { - HCILP_Enable(TRUE); - } else if (strstr(ptr, "-DUT")) { - EnableTestMode(); - } else if (strstr(ptr, "-FEATURE")) { - SetLocalFeatures(); - } else if (strstr(ptr, "-ATTACH")) { - EnbleHCI(); - while (1) { - sleep(UINT_MAX); - } - } else if (strstr(ptr, "-DEBUG")) { - } else if (strstr(ptr, "-CSTOPB")) { - - } else { - fprintf(stderr, "Invalid parameter(s)!\n"); - exit_err(1); - } - } - - fprintf(stderr, "EXIT\n"); - close(fd); - exit(0); - - return 0; -} -- 2.7.4