/* * Copyright 2010-2011 Christian Lamparter * * 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 version 2 of the License. * * 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 Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include #include #include #include #include #include #include #include #include "carlfw.h" #include "compiler.h" static int get_val(char *str, unsigned int *val) { int err; err = sscanf(str, "%8x", val); if (err != 1) return -EINVAL; return 0; } static int get_addr(char *str, unsigned int *val) { int err; err = get_val(str, val); if (*val & 3) { fprintf(stderr, "Address 0x%.8x is not a multiple of 4.\n", *val); return -EINVAL; } return err; } static int new_fix_entry(struct carlfw *fw, struct carl9170fw_fix_entry *fix_entry) { struct carl9170fw_fix_desc *fix; unsigned int len; len = sizeof(*fix) + sizeof(*fix_entry); fix = malloc(len); if (!fix) return -ENOMEM; carl9170fw_fill_desc(&fix->head, (uint8_t *) FIX_MAGIC, cpu_to_le16(len), CARL9170FW_FIX_DESC_MIN_VER, CARL9170FW_FIX_DESC_CUR_VER); memcpy(&fix->data[0], fix_entry, sizeof(*fix_entry)); return carlfw_desc_add_tail(fw, &fix->head); } static struct carl9170fw_fix_entry * scan_for_similar_fix(struct carl9170fw_fix_desc *fix, __le32 address) { unsigned int i, entries; entries = (le16_to_cpu(fix->head.length) - sizeof(*fix)) / sizeof(struct carl9170fw_fix_entry); for (i = 0; i < entries; i++) { if (address == fix->data[i].address) return &fix->data[i]; } return NULL; } static int add_another_fix_entry(struct carlfw *fw, struct carl9170fw_fix_desc *fix, struct carl9170fw_fix_entry *fix_entry) { unsigned int entry; fix = carlfw_desc_mod_len(fw, &fix->head, sizeof(*fix_entry)); if (IS_ERR_OR_NULL(fix)) return (int) PTR_ERR(fix); entry = (le16_to_cpu(fix->head.length) - sizeof(*fix)) / sizeof(*fix_entry) - 1; memcpy(&fix->data[entry], fix_entry, sizeof(*fix_entry)); return 0; } static int update_entry(char option, struct carl9170fw_fix_entry *entry, struct carl9170fw_fix_entry *fix) { switch (option) { case '=': entry->mask = fix->mask; entry->value = fix->value; break; case 'O': entry->mask |= fix->mask; entry->value |= fix->value; break; case 'A': entry->mask &= fix->mask; entry->value &= fix->value; break; default: fprintf(stderr, "Unknown option: '%c'\n", option); return -EINVAL; } return 0; } static void user_education(void) { fprintf(stderr, "Usage:\n"); fprintf(stderr, "\teeprom_fix FW-FILE SWITCH [ADDRESS [VALUE MASK]]\n"); fprintf(stderr, "\nDescription:\n"); fprintf(stderr, "\tThis utility manage a set of overrides which " "commands the driver\n\tto load customized EEPROM' " "data for all specified addresses.\n"); fprintf(stderr, "\nParameters:\n"); fprintf(stderr, "\t'FW-FILE' = firmware file [basename]\n"); fprintf(stderr, "\t'SWITCH' = [=|d|D]\n"); fprintf(stderr, "\t | '=' => add/set value for address\n"); fprintf(stderr, "\t | 'D' => removes all EEPROM overrides\n"); fprintf(stderr, "\t * 'd' => removed override for 'address'\n"); fprintf(stderr, "\n\t'ADDRESS' = location of the EEPROM override\n"); fprintf(stderr, "\t\t NB: must be a multiple of 4.\n"); fprintf(stderr, "\t\t an address map can be found in eeprom.h.\n"); fprintf(stderr, "\n\t'VALUE' = replacement value\n"); fprintf(stderr, "\t'MASK' = mask for the value placement.\n\n"); exit(EXIT_FAILURE); } static int set_fix(struct carlfw *fw, struct carl9170fw_fix_desc *fix, char __unused option, int __unused argc, char *args[]) { struct carl9170fw_fix_entry fix_entry, *entry = NULL; unsigned int address, value, mask; int err; err = get_addr(args[3], &address); if (err) return err; err = get_val(args[4], &value); if (err) return err; err = get_val(args[5], &mask); if (err) return err; fix_entry.address = cpu_to_le32(address); fix_entry.value = cpu_to_le32(value); fix_entry.mask = cpu_to_le32(mask); if (!fix) { err = new_fix_entry(fw, &fix_entry); } else { entry = scan_for_similar_fix(fix, fix_entry.address); if (entry) { err = update_entry(option, entry, &fix_entry); if (err) fprintf(stdout, "Overwrite old entry.\n"); } else { err = add_another_fix_entry(fw, fix, &fix_entry); } } return err; } static int del_fix(struct carlfw *fw, struct carl9170fw_fix_desc *fix, char __unused option, int __unused argc, char *args[]) { struct carl9170fw_fix_entry *entry = NULL; unsigned int address; unsigned long off; unsigned int rem_len; int err; err = get_addr(args[3], &address); if (err) return err; if (fix) entry = scan_for_similar_fix(fix, cpu_to_le32(address)); if (!entry) { fprintf(stderr, "Entry for 0x%.8x not found\n", address); return -EINVAL; } off = (unsigned long) entry - (unsigned long) fix->data; rem_len = le16_to_cpu(fix->head.length) - off; if (rem_len) { unsigned long cont; cont = (unsigned long) entry + sizeof(*entry); memmove(entry, (void *)cont, rem_len); } fix = carlfw_desc_mod_len(fw, &fix->head, -sizeof(*entry)); err = IS_ERR_OR_NULL(fix); return err; } static int del_all(struct carlfw *fw, struct carl9170fw_fix_desc *fix, char __unused option, int __unused argc, char __unused *args[]) { if (!fix) return 0; carlfw_desc_del(fw, &fix->head); return 0; } static const struct { char option; int argc; int (*func)(struct carlfw *, struct carl9170fw_fix_desc *, char, int, char **); } programm_function[] = { { '=', 6, set_fix }, { 'O', 6, set_fix }, { 'A', 6, set_fix }, { 'd', 4, del_fix }, { 'D', 3, del_all }, }; int main(int argc, char *args[]) { struct carl9170fw_fix_desc *fix; struct carlfw *fw = NULL; unsigned int i; int err = 0; char option; if (argc < 3 || argc > 6) { err = -EINVAL; goto out; } fw = carlfw_load(args[1]); if (IS_ERR_OR_NULL(fw)) { err = PTR_ERR(fw); fprintf(stderr, "Failed to open file \"%s\" (%d).\n", args[1], err); goto out; } fix = carlfw_find_desc(fw, (uint8_t *)FIX_MAGIC, sizeof(*fix), CARL9170FW_FIX_DESC_CUR_VER); option = args[2][0]; for (i = 0; i < ARRAY_SIZE(programm_function); i++) { if (programm_function[i].option != option) continue; if (argc != programm_function[i].argc) { err = -EINVAL; goto out; } err = programm_function[i].func(fw, fix, option, argc, args); if (err) goto out; break; } if (i == ARRAY_SIZE(programm_function)) { fprintf(stderr, "Unknown option: '%c'\n", args[2][0]); goto out; } err = carlfw_store(fw); if (err) { fprintf(stderr, "Failed to apply changes (%d).\n", err); goto out; } out: carlfw_release(fw); if (err) { if (err == -EINVAL) user_education(); else fprintf(stderr, "%s\n", strerror(err)); } return err ? EXIT_FAILURE : EXIT_SUCCESS; }