Diff between d80f68286c72a84fb77235f4753adcb59eadb44f and 2c7190715e2c3c5540591d477c6170245d1fd7a7

Changed Files

File Additions Deletions Status
.gitignore +1 -0 modified
Makefile.tools +4 -1 modified
tools/rtlfw.c +192 -0 added

Full Patch

diff --git a/.gitignore b/.gitignore
index 393735e..4780805 100644
--- a/.gitignore
+++ b/.gitignore
@@ -69,6 +69,7 @@ tools/bluemoon
 tools/seq2bseq
 tools/hex2hcd
 tools/nokfw
+tools/rtlfw
 tools/btiotest
 tools/mpris-proxy
 tools/bluetooth-player
diff --git a/Makefile.tools b/Makefile.tools
index b7b4225..651ff00 100644
--- a/Makefile.tools
+++ b/Makefile.tools
@@ -185,7 +185,8 @@ noinst_PROGRAMS += tools/bdaddr tools/avinfo tools/avtest \
 			tools/btsnoop tools/btproxy \
 			tools/btiotest tools/bneptest tools/mcaptest \
 			tools/cltest tools/oobtest tools/advtest \
-			tools/seq2bseq tools/nokfw tools/create-image \
+			tools/seq2bseq tools/nokfw tools/rtlfw \
+			tools/create-image \
 			tools/eddystone tools/ibeacon \
 			tools/btgatt-client tools/btgatt-server \
 			tools/test-runner tools/check-selftest \
@@ -249,6 +250,8 @@ tools_seq2bseq_SOURCES = tools/seq2bseq.c
 
 tools_nokfw_SOURCES = tools/nokfw.c
 
+tools_rtlfw_SOURCES = tools/rtlfw.c
+
 tools_create_image_SOURCES = tools/create-image.c
 
 tools_eddystone_SOURCES = tools/eddystone.c monitor/bt.h
diff --git a/tools/rtlfw.c b/tools/rtlfw.c
new file mode 100644
index 0000000..5e354a3
--- /dev/null
+++ b/tools/rtlfw.c
@@ -0,0 +1,192 @@
+/*
+ *
+ *  BlueZ - Bluetooth protocol stack for Linux
+ *
+ *  Copyright (C) 2012-2013  Intel Corporation
+ *
+ *
+ *  This program is free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 2 of the License, or
+ *  (at your option) any later version.
+ *
+ *  This program is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, write to the Free Software
+ *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <stdio.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <unistd.h>
+#include <string.h>
+#include <getopt.h>
+#include <stdlib.h>
+#include <stdint.h>
+#include <sys/stat.h>
+#include <sys/mman.h>
+
+#include <src/shared/util.h>
+
+#define CONFIG_MAGIC	0x8723ab55
+
+static const char *offset_to_str(uint16_t offset)
+{
+	switch (offset) {
+	case 0x00f4:
+		return "PCM_SETTING";
+	case 0x000c:
+		return "UART_CONFIG";
+	case 0x003c:
+		return "BD_ADDR";
+	}
+
+	return NULL;
+}
+
+static void analyze_memory(const uint8_t *buf, size_t len)
+{
+	const uint8_t *ptr = buf;
+	uint32_t magic;
+	uint16_t datalen;
+
+	if (len < 6) {
+		fprintf(stderr, "Invalid file length of %zu bytes\n", len);
+		return;
+	}
+
+	magic = get_le32(ptr);
+	datalen = get_le16(ptr + 4);
+
+	printf("Signature: 0x%8.8x\n", magic);
+	printf("Data len:  %u\n", datalen);
+
+	if (magic != CONFIG_MAGIC) {
+		fprintf(stderr, "Unsupported file signature\n");
+		return;
+	}
+
+	ptr += 6;
+
+	while (ptr < buf + datalen + 6) {
+		uint16_t offset;
+		uint8_t plen;
+		const char *str;
+		unsigned int i;
+
+		offset = get_le16(ptr);
+		plen = get_u8(ptr + 2);
+
+		if (ptr + plen + 3 > buf + datalen + 6) {
+			fprintf(stderr, "Invalid config entry size\n");
+			break;
+		}
+
+		str = offset_to_str(offset);
+
+		printf("len=%-3u offset=%4.4x,{ ", plen, offset);
+		for (i = 0; i < plen; i++)
+			printf("%2.2x ", ptr[3 + i]);
+		printf("}%s%s\n", str ? "," : "", str ? : "");
+
+		ptr += plen + 3;
+	}
+}
+
+static void analyze_file(const char *pathname)
+{
+	struct stat st;
+	void *map;
+	int fd;
+
+	printf("Analyzing %s\n", pathname);
+
+	fd = open(pathname, O_RDONLY | O_CLOEXEC);
+	if (fd < 0) {
+		perror("Failed to open file");
+		return;
+	}
+
+	if (fstat(fd, &st) < 0) {
+		fprintf(stderr, "Failed get file size\n");
+		close(fd);
+		return;
+	}
+
+	if (st.st_size == 0) {
+		fprintf(stderr, "Empty file\n");
+		close(fd);
+		return;
+	}
+
+	map = mmap(NULL, st.st_size, PROT_READ, MAP_SHARED, fd, 0);
+	if (!map || map == MAP_FAILED) {
+		fprintf(stderr, "Failed to map file\n");
+		close(fd);
+		return;
+        }
+
+	analyze_memory(map, st.st_size);
+
+	munmap(map, st.st_size);
+	close(fd);
+}
+
+static void usage(void)
+{
+	printf("Realtek Bluetooth firmware analyzer\n"
+		"Usage:\n");
+	printf("\trtlfw [options] <file>\n");
+	printf("Options:\n"
+		"\t-h, --help             Show help options\n");
+}
+
+static const struct option main_options[] = {
+	{ "version", no_argument,       NULL, 'v' },
+	{ "help",    no_argument,       NULL, 'h' },
+	{ }
+};
+
+int main(int argc, char *argv[])
+{
+	int i;
+
+	for (;;) {
+		int opt;
+
+		opt = getopt_long(argc, argv, "vh", main_options, NULL);
+		if (opt < 0)
+			break;
+
+		switch (opt) {
+		case 'v':
+			printf("%s\n", VERSION);
+			return EXIT_SUCCESS;
+		case 'h':
+			usage();
+			return EXIT_SUCCESS;
+		default:
+			return EXIT_FAILURE;
+		}
+	}
+
+	if (argc - optind < 1) {
+		fprintf(stderr, "No input firmware files provided\n");
+		return EXIT_FAILURE;
+	}
+
+	for (i = optind; i < argc; i++)
+		analyze_file(argv[i]);
+
+	return EXIT_SUCCESS;
+}