From ea49719f6c696ccd47fd01fd2583df53aaf26e99 Mon Sep 17 00:00:00 2001 From: DoHyun Pyun Date: Thu, 7 May 2015 11:01:40 +0900 Subject: [PATCH 2/9] Add BT firmware for Tizen mobile profile Change-Id: Ib625a6a1127925e70cb72d36c7a06dc28c2baebb Signed-off-by: DoHyun Pyun --- CMakeLists.txt | 15 + LICENSE | 204 ++++ debian/bluetooth-firmware-bcm.install | 2 + debian/bluetooth-firmware-bcm.postinst.in | 10 + debian/changelog | 23 + debian/compat | 1 + debian/control | 25 + debian/rules | 112 +++ firmware/BCM4330B1_002.001.003.0221.0265.hcd | Bin 0 -> 32195 bytes firmware/BT_FW_BCM4358A1_001.002.005.0032.0066.hcd | Bin 0 -> 66667 bytes firmware/CMakeLists.txt | 3 + packaging/bluetooth-firmware-bcm.spec | 39 + scripts/CMakeLists.txt | 4 + scripts/bt-dev-end.sh | 32 + scripts/bt-dev-start.sh | 57 ++ scripts/bt-set-addr.sh | 16 + set-address/CMakeLists.txt | 24 + set-address/setbd.c | 413 ++++++++ tools/CMakeLists.txt | 29 + tools/bcmtool_4330b1.c | 865 ++++++++++++++++ tools/bcmtool_4330b1_m.c | 1034 ++++++++++++++++++++ tools/bcmtool_4358a1.c | 865 ++++++++++++++++ 22 files changed, 3773 insertions(+) create mode 100644 CMakeLists.txt create mode 100644 LICENSE create mode 100644 debian/bluetooth-firmware-bcm.install create mode 100644 debian/bluetooth-firmware-bcm.postinst.in create mode 100755 debian/changelog create mode 100644 debian/compat create mode 100644 debian/control create mode 100755 debian/rules create mode 100644 firmware/BCM4330B1_002.001.003.0221.0265.hcd create mode 100755 firmware/BT_FW_BCM4358A1_001.002.005.0032.0066.hcd create mode 100755 firmware/CMakeLists.txt create mode 100644 packaging/bluetooth-firmware-bcm.spec create mode 100755 scripts/CMakeLists.txt create mode 100755 scripts/bt-dev-end.sh create mode 100755 scripts/bt-dev-start.sh create mode 100755 scripts/bt-set-addr.sh create mode 100644 set-address/CMakeLists.txt create mode 100644 set-address/setbd.c create mode 100644 tools/CMakeLists.txt create mode 100644 tools/bcmtool_4330b1.c create mode 100644 tools/bcmtool_4330b1_m.c create mode 100644 tools/bcmtool_4358a1.c diff --git a/CMakeLists.txt b/CMakeLists.txt new file mode 100644 index 0000000..c0285fa --- /dev/null +++ b/CMakeLists.txt @@ -0,0 +1,15 @@ +CMAKE_MINIMUM_REQUIRED(VERSION 2.6) + +INCLUDE(FindPkgConfig) +pkg_check_modules(package REQUIRED glib-2.0 vconf) + +FOREACH(flag ${package_CFLAGS}) + SET(EXTRA_CFLAGS "${EXTRA_CFLAGS} ${flag}") +ENDFOREACH(flag) + +SET(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${EXTRA_CFLAGS}") + +ADD_SUBDIRECTORY(set-address) +ADD_SUBDIRECTORY(tools) +ADD_SUBDIRECTORY(firmware) +ADD_SUBDIRECTORY(scripts) diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..a06208b --- /dev/null +++ b/LICENSE @@ -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/debian/bluetooth-firmware-bcm.install b/debian/bluetooth-firmware-bcm.install new file mode 100644 index 0000000..a99d1b4 --- /dev/null +++ b/debian/bluetooth-firmware-bcm.install @@ -0,0 +1,2 @@ +usr/bin/* +usr/etc/bluetooth/BCM4330B1_002.001.003.0221.0265.hcd diff --git a/debian/bluetooth-firmware-bcm.postinst.in b/debian/bluetooth-firmware-bcm.postinst.in new file mode 100644 index 0000000..cb03b3c --- /dev/null +++ b/debian/bluetooth-firmware-bcm.postinst.in @@ -0,0 +1,10 @@ +#!/bin/sh + +# Change File Permission +chmod 644 @PREFIX@/etc/bluetooth/BCM4330B1_002.001.003.0221.0265.hcd + +# root case +if [ ${USER} = "root" ] +then + chown root:root @PREFIX@/etc/bluetooth/BCM4330B1_002.001.003.0221.0265.hcd +fi diff --git a/debian/changelog b/debian/changelog new file mode 100755 index 0000000..fb08aec --- /dev/null +++ b/debian/changelog @@ -0,0 +1,23 @@ +bluetooth-firmware-bcm (0.1.3) unstable; urgency=low + + * rename bt-firmware-43xx to bluetooth-firmware-bcm + + -- Jaekyun Lee Sat, 11 Aug 2012 14:26:05 +0900 + +bt-firmware-43xx (0.1.2) unstable; urgency=low + + * Fix set bd bug as TAPI IMEI API deprecated + + -- Jaekyun Lee Mon, 26 Mar 2012 15:13:31 +0900 + +bt-firmware-43xx (0.1.1) unstable; urgency=low + + * common script moved to bluetooth-tools + + -- Jaekyun Lee Fri, 23 Mar 2012 11:29:47 +0900 + +bt-firmware-43xx (0.1.0) unstable; urgency=low + + * Initail Release + + -- DoHyun Pyun Tue, 14 Feb 2012 15:13:03 +0900 diff --git a/debian/compat b/debian/compat new file mode 100644 index 0000000..7ed6ff8 --- /dev/null +++ b/debian/compat @@ -0,0 +1 @@ +5 diff --git a/debian/control b/debian/control new file mode 100644 index 0000000..91d830a --- /dev/null +++ b/debian/control @@ -0,0 +1,25 @@ +Source: bluetooth-firmware-bcm +Section: devel +Priority: optional +Maintainer: DoHyun Pyun , ChanYeol Park , Girish A J , InJun Yang , Jaekyun Lee +Uploaders: Sunil Behera , Syam Sidhardhan +Build-Depends: debhelper (>= 5), libglib2.0-dev, libvconf-dev +Standards-Version: 3.7.2 +Homepage: N/A + +Package: bluetooth-firmware-bcm +Architecture: any +Section: utils +Depends: ${shlibs:Depends}, ${misc:Depends}, bluetooth-tools +Description: firmware and tools + firmware and tools + +Package: bluetooth-firmware-bcm-dbg +Section: debug +Architecture: any +Depends: ${shlibs:Depends}, ${misc:Depends}, bluetooth-firmware-bcm (= ${Source-Version}) +Description: Tools for bluetooth-firmware-bcm (unstripped) + The package contains detached debugging symbols for the binary packages + produced by the bt-firmware-43xx soruce. + + diff --git a/debian/rules b/debian/rules new file mode 100755 index 0000000..531d1a1 --- /dev/null +++ b/debian/rules @@ -0,0 +1,112 @@ +#!/usr/bin/make -f + +# Uncomment this to turn on verbose mode. +#export DH_VERBOSE=1 + +CFLAGS ?= -Wall -g +LDFLAGS ?= +PREFIX ?= /usr +DATADIR ?= /opt +PLUGIN_PREFIX ?= /usr + +CMAKE_BUILD_DIR ?= $(CURDIR)/cmake_build_tmp + +ifneq (,$(findstring noopt,$(DEB_BUILD_OPTIONS))) + CFLAGS += -O0 +else + CFLAGS += -O2 +endif + +LDFLAGS += -Wl,--rpath=$(PREFIX)/lib -Wl,--as-needed -Wl,--unresolved-symbols=ignore-in-shared-libs + +configure: configure-stamp +configure-stamp: + dh_testdir + # Add here commands to configure the package. + mkdir -p $(CMAKE_BUILD_DIR) && \ + cd $(CMAKE_BUILD_DIR) && \ + CFLAGS="$(CFLAGS) -fpie" LDFLAGS="$(LDFLAGS) -pie" \ + cmake .. -DCMAKE_INSTALL_PREFIX=$(PREFIX) -DPLUGIN_INSTALL_PREFIX=$(PLUGIN_PREFIX) + + touch configure-stamp + +build: build-stamp + +build-stamp: configure-stamp + dh_testdir + + # Add here commands to compile the package. + cd $(CMAKE_BUILD_DIR) && $(MAKE) + + for f in `find $(CURDIR)/debian/ -name "*.in"`; do \ + cat $$f > $${f%.in}; \ + sed -i -e "s#@PREFIX@#$(PREFIX)#g" $${f%.in}; \ + sed -i -e "s#@DATADIR@#$(DATADIR)#g" $${f%.in}; \ + sed -i -e "s#@PLUGIN_PREFIX@#$(PLUGIN_PREFIX)#g" $${f%.in}; \ + done + + touch $@ + +clean: + dh_testdir + dh_testroot + rm -f build-stamp configure-stamp + + # Add here commands to clean up after the build process. + -rm -rf $(CMAKE_BUILD_DIR) + for f in `find $(CURDIR)/debian/ -name "*.in"`; do \ + rm -f $${f%.in}; \ + done + + dh_clean + +install: build + dh_testdir + dh_testroot + dh_clean -k + dh_installdirs + + # Add here commands to install the package into debian/wavplayer. + cd $(CMAKE_BUILD_DIR) && $(MAKE) DESTDIR=$(CURDIR)/debian/tmp install + + + + # Booting script + +# Build architecture-independent files here. +binary-indep: build install +# We have nothing to do by default. + +# Build architecture-dependent files here. +binary-arch: build install + dh_testdir + dh_testroot + dh_installchangelogs + dh_installdocs + dh_installexamples + dh_install --sourcedir=debian/tmp +# dh_installmenu +# dh_installdebconf +# dh_installlogrotate +# dh_installemacsen +# dh_installpam +# dh_installmime +# dh_python +# dh_installinit +# dh_installcron +# dh_installinfo + dh_installman + dh_link + dh_strip --dbg-package=bluetooth-firmware-bcm-dbg + dh_compress + dh_fixperms +# dh_perl + dh_makeshlibs + dh_installdeb + dh_shlibdeps + dh_gencontrol + dh_md5sums + dh_builddeb + +binary: binary-indep binary-arch +.PHONY: build clean binary-indep binary-arch binary install configure diff --git a/firmware/BCM4330B1_002.001.003.0221.0265.hcd b/firmware/BCM4330B1_002.001.003.0221.0265.hcd new file mode 100644 index 0000000000000000000000000000000000000000..a8c7f302ed86f84eedbd65a64a9947947dcb2f0f GIT binary patch literal 32195 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 T5|+thN-__Poi$?y%5U_48~_0} literal 0 HcmV?d00001 diff --git a/firmware/BT_FW_BCM4358A1_001.002.005.0032.0066.hcd b/firmware/BT_FW_BCM4358A1_001.002.005.0032.0066.hcd new file mode 100755 index 0000000000000000000000000000000000000000..d5d7795f86a761036475dd6a474700a3dc294622 GIT binary patch literal 66667 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 /sys/class/gpio/gpio17/value +# sleep 0.1 +# echo 1 > /sys/class/gpio/gpio17/value +#fi + diff --git a/scripts/bt-dev-start.sh b/scripts/bt-dev-start.sh new file mode 100755 index 0000000..42d4d3b --- /dev/null +++ b/scripts/bt-dev-start.sh @@ -0,0 +1,57 @@ +#!/bin/sh + +# Script for registering Broadcom UART BT device +BT_UART_DEVICE=/dev/ttySAC3 +BT_CHIP_TYPE=bcm2035 +#BCM_TOOL=/usr/bin/bcmtool_4330b1 +BCM_TOOL=/usr/bin/bcmtool_4358a1 + +BT_PLATFORM_DEFAULT_HCI_NAME="TIZEN-Mobile" +UART_SPEED=3000000 + +#set default firmware +#BCM_FIRMWARE=BT_FW_BCM4330B1_002.001.003.0221.0265.hcd +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}"` + +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 +rfkill unblock bluetooth + +echo "Check for Bluetooth device status" +if (/usr/sbin/hciconfig | grep hci); then + echo "Bluetooth device is UP" + /usr/sbin/hciconfig hci0 up +else + echo "Bluetooth device is DOWN" + echo "Registering Bluetooth device" + + $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 + sleep 0.1 + /usr/sbin/hciconfig hci0 up + /usr/sbin/hciconfig hci0 name $BT_PLATFORM_DEFAULT_HCI_NAME + /usr/sbin/hciconfig hci0 sspmode 1 + echo "HCIATTACH success" + else + echo "HCIATTACH failed" + rfkill block bluetooth + fi +fi diff --git a/scripts/bt-set-addr.sh b/scripts/bt-set-addr.sh new file mode 100755 index 0000000..57b2d02 --- /dev/null +++ b/scripts/bt-set-addr.sh @@ -0,0 +1,16 @@ +#!/bin/sh + +# +# Script for setting Bluetooth Address +# + +if [ -e /opt/etc/.bd_addr ] +then + echo "Already .bd_addr exists" + exit 0 +fi + +/usr/bin/setbd + +echo "Set BT address successes" + diff --git a/set-address/CMakeLists.txt b/set-address/CMakeLists.txt new file mode 100644 index 0000000..2ff386b --- /dev/null +++ b/set-address/CMakeLists.txt @@ -0,0 +1,24 @@ +CMAKE_MINIMUM_REQUIRED(VERSION 2.6) +PROJECT(setbd C) + +SET(SRCS setbd.c) + +SET(PREFIX ${CMAKE_INSTALL_PREFIX}) + +FIND_PROGRAM(UNAME NAMES uname) +EXEC_PROGRAM("${UNAME}" ARGS "-m" OUTPUT_VARIABLE "ARCH") +IF("${ARCH}" STREQUAL "arm") + ADD_DEFINITIONS("-DTARGET") + MESSAGE("add -DTARGET") +ENDIF("${ARCH}" STREQUAL "arm") + +ADD_DEFINITIONS("-DPREFIX=\"${CMAKE_INSTALL_PREFIX}\"") +ADD_DEFINITIONS("-DFACTORYFS=\"$ENV{FACTORYFS}\"") +ADD_DEFINITIONS("-D__BROADCOM_PATCH__") +ADD_DEFINITIONS("-DDEBUG_EN") + +ADD_EXECUTABLE(${PROJECT_NAME} ${SRCS}) +TARGET_LINK_LIBRARIES(${PROJECT_NAME} ${package_LDFLAGS}) + +INSTALL(TARGETS ${PROJECT_NAME} DESTINATION bin) + diff --git a/set-address/setbd.c b/set-address/setbd.c new file mode 100644 index 0000000..092da16 --- /dev/null +++ b/set-address/setbd.c @@ -0,0 +1,413 @@ +/* + * setbd.c + * + * Copyright (c) 2000 - 2011 Samsung Electronics Co., Ltd. All rights reserved + * + * Contact: Hocheol Seo + * GirishAshok Joshi + * DoHyun Pyun + * + * 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 + +#ifdef __TI_PATCH__ +#define BT_CHIP_TI +#else +#ifdef __BROADCOM_PATCH__ +#define BT_CHIP_BROADCOM +#else +#define BT_CHIP_CSR +#endif +#endif + +#ifdef DEBUG_EN +#define APP_DBG(format, args...) printf("%s(), line[%d]: " format, __FUNCTION__, __LINE__, ##args) +#define APP_DEBUG(format, args...) printf(format, ##args) +#else +#define APP_DBG(format, args...) +#define APP_DEBUG(format, args...) +#endif + +#define BD_ADDR_FILE "/opt/etc/.bd_addr" +#define PSKEY_TEMP_FILE "/opt/etc/.bluetooth.psr" +#define PSR_FILE "/opt/etc/bluetooth/bluetooth.psr" + +#define BD_ADDR_LEN 14 +#define BD_PREFIX "0002\n" + +#define PSKEY_LEN 27 +#define PSKEY_BDADDR_PREFIX "&0001 = 0012 " + +#define READ_BD_FILE_MAX 300 + +static int success_make_bt_address_from_imei=-1; +static GMainLoop * loop; +const char *DEFAULT_IMEI="004999010640000"; +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; + + fd = fopen(path, "r"); + if(NULL == fd){ + APP_DBG("Error open psr file\r\n"); + return -21; + } + + new = fopen(PSKEY_TEMP_FILE, "w"); + if(NULL == new){ + APP_DBG("Error creat temp file\r\n"); + fclose(fd); + return -22; + } + + ret = fputs(pskey,new); + + while(1){ + result = fgets(cmp, READ_BD_FILE_MAX, fd); + + APP_DBG("PSR : [%s]\r\n", cmp); + + if((NULL != result) && (0 == strncmp(cmp, "&0001", 5))){ + APP_DBG("Find BD address set script\r\n"); + continue; + } + + if(NULL == result){ + APP_DBG("EOF reaches\r\n"); + fclose(fd); + fclose(new); + return 1; + } + + ret = fputs(cmp,new); + } + + return 0; +} +void makeRandomBD(unsigned char* buf){ + int ran; + int i; + unsigned int seed; + memcpy(buf, BD_PREFIX, 5); + seed = time(NULL); + for(i=5;i<14;i++){ + if(7==i){ + buf[i]='\n'; + continue; + } + ran = rand_r(&seed)%16; + if(10>ran) + ran+=0x30; + else + ran+=0x57; + buf[i]=ran; + } + APP_DEBUG("Random number is\r\n"); + for(i=0;ihci_read_bd_addr(fd, &bdaddr, 1000)){ + APP_DBG("Read BD ADDR failed!!!\r\n"); + return -3; + } + hci_close_dev(fd); + + ba2str(&bdaddr, address); + for(i=0;i<17;i++){ + if(':' == address[i]) + continue; + + if(5>i) + nap[cnt_nap++] = address[i]; + else if(8>i) + uap[cnt_uap++] = address[i]; + else + lap[cnt_lap++] = address[i]; + } + + APP_DBG("BT address [%s], nap[%c%c%c%c], uap[%c%c], lap[%c%c%c%c%c%c]\r\n",\ + address, nap[0], nap[1], nap[2], nap[3]\ + , uap[0],uap[1]\ + ,lap[0], lap[1],lap[2],lap[3],lap[4],lap[5]); + + + filedesc=open(BD_ADDR_FILE, O_RDWR | O_CREAT | O_TRUNC | O_SYNC, 0644); + if(0>filedesc){ + APP_DBG("File creation fail!!!\r\n"); + return -4; + } + ret = write(filedesc, nap, 4); + ret = write(filedesc, "\n", 1); + ret = write(filedesc, uap, 2); + ret = write(filedesc, "\n", 1); + ret = write(filedesc, lap, 6); + ret = write(filedesc, "\n", 1); + close(filedesc); + + 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;ifd){ + APP_DBG("File not exists\r\n"); + ret=readBDaddrTI(); + }else{ + APP_DBG("File exists\r\n"); + close(fd); + ret=0; + } + + return ret; +#else + printf("error BT CHIP not defined!!!\n"); + return 0; +#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) +{ + + APP_DEBUG("Time out!!!\n"); + g_main_loop_quit(loop); + + return FALSE; +} +int main() +{ + loop=g_main_loop_new(NULL,FALSE); + APP_DEBUG("Bluetooth Address Setting\n"); + make_bt_address(FALSE); + + /* When we get proper BT address by IMEI or + * randome BT address due to default IMEI + * We don't need to wait for telephony activation, timeout */ + if(success_make_bt_address_from_imei==0 || is_default_imei==TRUE) + 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; +} diff --git a/tools/CMakeLists.txt b/tools/CMakeLists.txt new file mode 100644 index 0000000..d37148e --- /dev/null +++ b/tools/CMakeLists.txt @@ -0,0 +1,29 @@ +CMAKE_MINIMUM_REQUIRED(VERSION 2.6) +PROJECT(bcmtool C) + +SET(SRCS_4330B1 bcmtool_4330b1.c) +SET(BCMTOOL_4330B1 ${PROJECT_NAME}_4330b1) + +SET(SRCS_4358A1 bcmtool_4358a1.c) +SET(BCMTOOL_4358A1 ${PROJECT_NAME}_4358a1) + +SET(PREFIX ${CMAKE_INSTALL_PREFIX}) + +FIND_PROGRAM(UNAME NAMES uname) +EXEC_PROGRAM("${UNAME}" ARGS "-m" OUTPUT_VARIABLE "ARCH") +IF("${ARCH}" STREQUAL "arm") + ADD_DEFINITIONS("-DTARGET") + MESSAGE("add -DTARGET") +ENDIF("${ARCH}" STREQUAL "arm") + +ADD_DEFINITIONS("-DPREFIX=\"${CMAKE_INSTALL_PREFIX}\"") +ADD_DEFINITIONS("-DFACTORYFS=\"$ENV{FACTORYFS}\"") + +ADD_EXECUTABLE(${BCMTOOL_4330B1} ${SRCS_4330B1}) +TARGET_LINK_LIBRARIES(${BCMTOOL_4330B1} ${package_LDFLAGS}) + +ADD_EXECUTABLE(${BCMTOOL_4358A1} ${SRCS_4358A1}) +TARGET_LINK_LIBRARIES(${BCMTOOL_4358A1} ${package_LDFLAGS}) + +# install binary file +INSTALL(TARGETS ${BCMTOOL_4358A1} DESTINATION ${PLUGIN_INSTALL_PREFIX}/bin) diff --git a/tools/bcmtool_4330b1.c b/tools/bcmtool_4330b1.c new file mode 100644 index 0000000..6350a06 --- /dev/null +++ b/tools/bcmtool_4330b1.c @@ -0,0 +1,865 @@ +/***************************************************************************** +** +** 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) + +/* 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 + +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_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 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 */ + +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 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 ( HIGH_SPEED_PATCHRAM_DOWNLOAD == TRUE ) + ChangeBaudRate(921600); +#endif + + 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); + DisplayProgress(FileSize,SentSize); + /* dispaly progress*/ + + read_event(fd, buffer); + + } + fclose(pFile); + + usleep(100000); /*100ms delay */ + + tcflush(fd, TCIOFLUSH); + tcgetattr(fd, &termios); + cfmakeraw(&termios); + termios.c_cflag |= CRTSCTS; + 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; + 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,"\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; + 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 +#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; +} diff --git a/tools/bcmtool_4358a1.c b/tools/bcmtool_4358a1.c new file mode 100644 index 0000000..b35da8e --- /dev/null +++ b/tools/bcmtool_4358a1.c @@ -0,0 +1,865 @@ +/***************************************************************************** +** +** 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) + +/* 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 + +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_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 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 */ + +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 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 ( HIGH_SPEED_PATCHRAM_DOWNLOAD == TRUE ) + ChangeBaudRate(921600); +#endif + + 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); + DisplayProgress(FileSize,SentSize); + /* dispaly progress*/ + + read_event(fd, buffer); + + } + fclose(pFile); + + usleep(100000); /*100ms delay */ + + tcflush(fd, TCIOFLUSH); + tcgetattr(fd, &termios); + cfmakeraw(&termios); + termios.c_cflag |= CRTSCTS; + 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; + 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,"\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; + 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 Date: Thu, 2 Jul 2015 17:31:13 +0900 Subject: [PATCH 3/9] 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 4/9] 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 5/9] 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 6/9] 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 7/9] [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 8/9] 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 9/9] 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