ar71xx: routerboot: add support for extended radio data

On newer Mikrotik boards, the radio calibration data
is stored differently and uses LZO compression instead
of RLE.

Update the RouterBOOT helper code to support the new
format.

Signed-off-by: Gabor Juhos <juhosg@openwrt.org>

SVN-Revision: 45297
This commit is contained in:
Gabor Juhos 2015-04-07 20:03:35 +00:00
parent f0cbd004ee
commit 76b5a9aa7d
4 changed files with 67 additions and 8 deletions

View file

@ -15,11 +15,13 @@
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/routerboot.h> #include <linux/routerboot.h>
#include <linux/rle.h> #include <linux/rle.h>
#include <linux/lzo.h>
#include "routerboot.h" #include "routerboot.h"
#define RB_BLOCK_SIZE 0x1000 #define RB_BLOCK_SIZE 0x1000
#define RB_ART_SIZE 0x10000 #define RB_ART_SIZE 0x10000
#define RB_MAGIC_ERD 0x00455244 /* extended radio data */
static struct rb_info rb_info; static struct rb_info rb_info;
@ -63,6 +65,7 @@ routerboot_find_tag(u8 *buf, unsigned int buflen, u16 tag_id,
u8 **tag_data, u16 *tag_len) u8 **tag_data, u16 *tag_len)
{ {
uint32_t magic; uint32_t magic;
bool align = false;
int ret; int ret;
if (buflen < 4) if (buflen < 4)
@ -70,6 +73,9 @@ routerboot_find_tag(u8 *buf, unsigned int buflen, u16 tag_id,
magic = get_u32(buf); magic = get_u32(buf);
switch (magic) { switch (magic) {
case RB_MAGIC_ERD:
align = true;
/* fall trough */
case RB_MAGIC_HARD: case RB_MAGIC_HARD:
/* skip magic value */ /* skip magic value */
buf += 4; buf += 4;
@ -121,6 +127,9 @@ routerboot_find_tag(u8 *buf, unsigned int buflen, u16 tag_id,
break; break;
} }
if (align)
len = (len + 3) / 4;
buf += len; buf += len;
buflen -= len; buflen -= len;
} }
@ -168,13 +177,16 @@ rb_get_hw_options(void)
return get_u32(tag); return get_u32(tag);
} }
__init void * static void * __init
rb_get_wlan_data(void) __rb_get_wlan_data(u16 id)
{ {
u16 tag_len; u16 tag_len;
u8 *tag; u8 *tag;
void *buf; void *buf;
int err; int err;
u32 magic;
size_t src_done;
size_t dst_done;
err = rb_find_hard_cfg_tag(RB_ID_WLAN_DATA, &tag, &tag_len); err = rb_find_hard_cfg_tag(RB_ID_WLAN_DATA, &tag, &tag_len);
if (err) { if (err) {
@ -188,12 +200,39 @@ rb_get_wlan_data(void)
goto err; goto err;
} }
magic = get_u32(tag);
if (magic == RB_MAGIC_ERD) {
u8 *erd_data;
u16 erd_len;
if (id == 0)
goto err_free;
err = routerboot_find_tag(tag, tag_len, id,
&erd_data, &erd_len);
if (err) {
pr_err("no ERD data found for id %u\n", id);
goto err_free;
}
dst_done = RB_ART_SIZE;
err = lzo1x_decompress_safe(erd_data, erd_len, buf, &dst_done);
if (err) {
pr_err("unable to decompress calibration data %d\n",
err);
goto err_free;
}
} else {
if (id != 0)
goto err_free;
err = rle_decode((char *) tag, tag_len, buf, RB_ART_SIZE, err = rle_decode((char *) tag, tag_len, buf, RB_ART_SIZE,
NULL, NULL); &src_done, &dst_done);
if (err) { if (err) {
pr_err("unable to decode calibration data\n"); pr_err("unable to decode calibration data\n");
goto err_free; goto err_free;
} }
}
return buf; return buf;
@ -203,6 +242,18 @@ err:
return NULL; return NULL;
} }
__init void *
rb_get_wlan_data(void)
{
return __rb_get_wlan_data(0);
}
__init void *
rb_get_ext_wlan_data(u16 id)
{
return __rb_get_wlan_data(id);
}
__init const struct rb_info * __init const struct rb_info *
rb_init_info(void *data, unsigned int size) rb_init_info(void *data, unsigned int size)
{ {

View file

@ -24,6 +24,7 @@ struct rb_info {
#ifdef CONFIG_ATH79_ROUTERBOOT #ifdef CONFIG_ATH79_ROUTERBOOT
const struct rb_info *rb_init_info(void *data, unsigned int size); const struct rb_info *rb_init_info(void *data, unsigned int size);
void *rb_get_wlan_data(void); void *rb_get_wlan_data(void);
void *rb_get_ext_wlan_data(u16 id);
int routerboot_find_tag(u8 *buf, unsigned int buflen, u16 tag_id, int routerboot_find_tag(u8 *buf, unsigned int buflen, u16 tag_id,
u8 **tag_data, u16 *tag_len); u8 **tag_data, u16 *tag_len);
@ -40,6 +41,11 @@ static inline void *rb_get_wlan_data(void)
return NULL; return NULL;
} }
static inline void *rb_get_wlan_data(u16 id)
{
return NULL;
}
static inline int static inline int
routerboot_find_tag(u8 *buf, unsigned int buflen, u16 tag_id, routerboot_find_tag(u8 *buf, unsigned int buflen, u16 tag_id,
u8 **tag_data, u16 *tag_len) u8 **tag_data, u16 *tag_len)

View file

@ -124,6 +124,7 @@ CONFIG_GPIO_74X164=y
CONFIG_GPIO_LATCH=y CONFIG_GPIO_LATCH=y
# CONFIG_JFFS2_FS is not set # CONFIG_JFFS2_FS is not set
CONFIG_LEDS_RB750=y CONFIG_LEDS_RB750=y
CONFIG_LZO_DECOMPRESS=y
CONFIG_MDIO_BITBANG=y CONFIG_MDIO_BITBANG=y
CONFIG_MDIO_GPIO=y CONFIG_MDIO_GPIO=y
# CONFIG_MTD_CFI is not set # CONFIG_MTD_CFI is not set

View file

@ -1379,11 +1379,12 @@
def_bool n def_bool n
config ATH79_DEV_GPIO_BUTTONS config ATH79_DEV_GPIO_BUTTONS
@@ -154,6 +1226,10 @@ config ATH79_PCI_ATH9K_FIXUP @@ -154,6 +1226,11 @@ config ATH79_PCI_ATH9K_FIXUP
def_bool n def_bool n
config ATH79_ROUTERBOOT config ATH79_ROUTERBOOT
+ select RLE_DECOMPRESS + select RLE_DECOMPRESS
+ select LZO_DECOMPRESS
+ def_bool n + def_bool n
+ +
+config PCI_AR724X +config PCI_AR724X