K2LL33D SHELL

 Apache/2.4.7 (Ubuntu)
 Linux sman1baleendah 3.13.0-24-generic #46-Ubuntu SMP Thu Apr 10 19:11:08 UTC 2014 x86_64
 uid=33(www-data) gid=33(www-data) groups=33(www-data)
 safemode : OFF
 MySQL: ON | Perl: ON | cURL: OFF | WGet: ON
  >  / lib / firmware / carl9170fw / tools / src /
server ip : 172.67.156.115

your ip : 172.70.80.135

H O M E


Filename/lib/firmware/carl9170fw/tools/src/eeprom_fix.c
Size7.32 kb
Permissionrw-r--r--
Ownerroot : root
Create time27-Apr-2025 09:50
Last modified05-Mar-2014 23:45
Last accessed06-Jul-2025 21:21
Actionsedit | rename | delete | download (gzip)
Viewtext | code | image
/*
* Copyright 2010-2011 Christian Lamparter <[email protected]>
*
* 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 <stdlib.h>
#include <stdio.h>
#include <error.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>

#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;
}