summaryrefslogtreecommitdiff
path: root/tools/rtlfw.c
diff options
context:
space:
mode:
authorMarcel Holtmann <marcel@holtmann.org>2018-01-05 21:46:38 +0100
committerMarcel Holtmann <marcel@holtmann.org>2018-01-05 21:46:38 +0100
commit2c7190715e2c3c5540591d477c6170245d1fd7a7 (patch)
tree1e931432e04d11a5549a19f686dde4e4edbc8584 /tools/rtlfw.c
parentd80f68286c72a84fb77235f4753adcb59eadb44f (diff)
downloadbluez-2c7190715e2c3c5540591d477c6170245d1fd7a7.tar.gz
tools: Add utility for Realtek config file parsing
Diffstat (limited to 'tools/rtlfw.c')
-rw-r--r--tools/rtlfw.c192
1 files changed, 192 insertions, 0 deletions
diff --git a/tools/rtlfw.c b/tools/rtlfw.c
new file mode 100644
index 000000000..5e354a3f5
--- /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;
+}