// SPDX-License-Identifier: GPL-2.0-or-later /* * * BlueZ - Bluetooth protocol stack for Linux * * Copyright (C) 2012-2013 Intel Corporation * * */ #ifdef HAVE_CONFIG_H #include #endif #define _GNU_SOURCE #include #include #include #include #include #include #include #include #include #include #include static ssize_t process_record(int fd, const char *line, uint16_t *upper_addr) { const char *ptr = line + 1; char str[3]; size_t len; uint8_t *buf; uint32_t addr; uint8_t sum = 0; int n = 0; if (line[0] != ':') { fprintf(stderr, "Invalid record start code (%c)\n", line[0]); return -EINVAL; } len = strlen(line); if (len < 11) { fprintf(stderr, "Record information is too short\n"); return -EILSEQ; } buf = malloc((len / 2) + 3); if (!buf) { fprintf(stderr, "Failed to allocate memory for record data\n"); return -ENOMEM; } while (1) { str[0] = *ptr++; str[1] = *ptr++; str[2] = '\0'; buf[3 + n] = strtol(str, NULL, 16); if (*ptr == '\r' || *ptr == '\n') break; sum += buf[3 + n++]; } sum = 0x100 - (sum & 0xff); if (n < 4 || buf[3] + 4 != n) { fprintf(stderr, "Record length is not matching data\n"); free(buf); return -EILSEQ; } if (buf[3 + n] != sum) { fprintf(stderr, "Checksum mismatch\n"); free(buf); return -EILSEQ; } switch (buf[6]) { case 0x00: addr = (*upper_addr << 16) + (buf[4] << 8) + buf[5]; buf[0] = 0x4c; buf[1] = 0xfc; buf[2] = n; buf[3] = (addr & 0x000000ff); buf[4] = (addr & 0x0000ff00) >> 8; buf[5] = (addr & 0x00ff0000) >> 16; buf[6] = (addr & 0xff000000) >> 24; if (write(fd, buf, n + 3) < 0) { perror("Failed to write data record"); free(buf); return -errno; } break; case 0x01: buf[0] = 0x4e; buf[1] = 0xfc; buf[2] = 0x04; buf[3] = 0xff; buf[4] = 0xff; buf[5] = 0xff; buf[6] = 0xff; if (write(fd, buf, 7) < 0) { perror("Failed to write end record"); free(buf); return -errno; } break; case 0x04: *upper_addr = (buf[7] << 8) + buf[8]; break; default: fprintf(stderr, "Unsupported record type (%02X)\n", buf[3]); free(buf); return -EILSEQ; } free(buf); return len; } static void convert_file(const char *input_path, const char *output_path) { uint16_t upper_addr = 0x0000; size_t line_size = 1024; char line_buffer[line_size]; char *path; const char *ptr; FILE *fp; struct stat st; off_t cur = 0; int fd; if (output_path) { path = strdup(output_path); if (!path) { perror("Failed to allocate string"); return; } } else { ptr = strrchr(input_path, '.'); if (ptr) { path = malloc(ptr - input_path + 6); if (!path) { perror("Failed to allocate string"); return; } strncpy(path, input_path, ptr - input_path); strcpy(path + (ptr - input_path), ".hcd"); } else { if (asprintf(&path, "%s.hcd", input_path) < 0) { perror("Failed to allocate string"); return; } } } printf("Converting %s to %s\n", input_path, path); fd = open(path, O_WRONLY | O_CREAT | O_TRUNC, 0644); free(path); if (fd < 0) { perror("Failed to create output file"); return; } if (stat(input_path, &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; } fp = fopen(input_path, "r"); if (!fp) { fprintf(stderr, "Failed to open input file\n"); close(fd); return; } while (1) { char *str; ssize_t len; str = fgets(line_buffer, line_size - 1, fp); if (!str) break; len = process_record(fd, str, &upper_addr); if (len < 0) goto done; cur += len; } if (cur != st.st_size) { fprintf(stderr, "Data length does not match file length\n"); goto done; } done: fclose(fp); close(fd); } struct ver_data { uint16_t num; char name[20]; char major[4]; char minor[4]; char build[4]; struct ver_data *next; }; static struct ver_data *ver_list = NULL; static void ver_parse_file(const char *pathname) { struct ver_data *ver, *tmp, *prev; char dummy1[5], dummy2[5]; if (strlen(pathname) < 7) return; if (strncmp(pathname, "BCM", 3)) return; ver = malloc(sizeof(*ver)); if (!ver) return; memset(ver, 0, sizeof(*ver)); if (sscanf(pathname, "%19[A-Z0-9]_%3c.%3c.%3c.%4c.%4c.hex", ver->name, ver->major, ver->minor, ver->build, dummy1, dummy2) != 6) { printf("\t/* failed to parse %s */\n", pathname); free(ver); return; } ver->num = atoi(ver->build) + (atoi(ver->minor) << 8) + (atoi(ver->major) << 13); if (!ver_list) { ver_list = ver; return; } for (tmp = ver_list, prev = NULL; tmp; prev = tmp, tmp = tmp->next) { if (ver->num == tmp->num) { free(ver); return; } if (ver->num < tmp->num) { if (prev) { prev->next = ver; ver->next = tmp; } else { ver->next = ver_list; ver_list = ver; } return; } } prev->next = ver; } static void ver_parse_entry(const char *pathname) { struct stat st; int fd; fd = open(pathname, O_RDONLY); if (fd < 0) { printf("\t/* failed to open %s */\n", pathname); return; } if (fstat(fd, &st) < 0) { printf("\t/* failed to stat %s */\n", pathname); goto done; } if (S_ISREG(st.st_mode)) { ver_parse_file(basename(pathname)); goto done; } if (S_ISDIR(st.st_mode)) { DIR *dir; dir = fdopendir(fd); if (!dir) goto done; while (1) { struct dirent *d; d = readdir(dir); if (!d) break; if (d->d_type == DT_REG) ver_parse_file(d->d_name); } closedir(dir); } done: close(fd); } static void ver_print_table(int argc, char *argv[]) { struct ver_data *ver; printf("static const struct {\n"); printf("\tuint16_t ver;\n"); printf("\tconst char *str\n"); printf("} table[] = {\n"); if (argc > 0) { int i; for (i = 0; i < argc; i++) ver_parse_entry(argv[i]); } else ver_parse_entry("."); for (ver = ver_list; ver; ) { struct ver_data *tmp = ver; printf("\t{ 0x%4.4x, \"%s\"\t},\t/* %s.%s.%s */\n", ver->num, ver->name, ver->major, ver->minor, ver->build); ver = ver->next; free(tmp); } printf(" { }\n"); printf("};\n"); } static void usage(void) { printf("Broadcom Bluetooth firmware converter\n" "Usage:\n"); printf("\thex2hcd [options] \n"); printf("Options:\n" "\t-o, --output Provide firmware output file\n" "\t-h, --help Show help options\n"); } static const struct option main_options[] = { { "table", no_argument, NULL, 'T' }, { "output", required_argument, NULL, 'o' }, { "version", no_argument, NULL, 'v' }, { "help", no_argument, NULL, 'h' }, { } }; int main(int argc, char *argv[]) { const char *output_path = NULL; bool print_table = false; int i; for (;;) { int opt; opt = getopt_long(argc, argv, "To:vh", main_options, NULL); if (opt < 0) break; switch (opt) { case 'T': print_table = true; break; case 'o': output_path = optarg; break; case 'v': printf("%s\n", VERSION); return EXIT_SUCCESS; case 'h': usage(); return EXIT_SUCCESS; default: return EXIT_FAILURE; } } if (print_table) { ver_print_table(argc - optind, argv + optind); return EXIT_SUCCESS; } if (argc - optind < 1) { fprintf(stderr, "No input firmware files provided\n"); return EXIT_FAILURE; } if (output_path && argc - optind > 1) { fprintf(stderr, "Only single input firmware supported\n"); return EXIT_FAILURE; } for (i = optind; i < argc; i++) convert_file(argv[i], output_path); return EXIT_SUCCESS; }