wl1251: make wl1251_set_partition bus agnostic
The same partition setting code can be used for both SPI and SDIO modes, if we remove the spi-specific commands and use the more generic buffer write routines. Do that and move it to io.c since it deals with register/memory address offsets. Signed-off-by: Bob Copeland <me@bobcopeland.com> Signed-off-by: Kalle Valo <kalle.valo@nokia.com> Signed-off-by: John W. Linville <linville@tuxdriver.com>
This commit is contained in:
parent
08d9f57251
commit
0d77e14133
|
@ -84,3 +84,98 @@ void wl1251_reg_write32(struct wl1251 *wl, int addr, u32 val)
|
||||||
{
|
{
|
||||||
wl1251_write32(wl, wl1251_translate_reg_addr(wl, addr), val);
|
wl1251_write32(wl, wl1251_translate_reg_addr(wl, addr), val);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Set the partitions to access the chip addresses.
|
||||||
|
*
|
||||||
|
* There are two VIRTUAL partitions (the memory partition and the
|
||||||
|
* registers partition), which are mapped to two different areas of the
|
||||||
|
* PHYSICAL (hardware) memory. This function also makes other checks to
|
||||||
|
* ensure that the partitions are not overlapping. In the diagram below, the
|
||||||
|
* memory partition comes before the register partition, but the opposite is
|
||||||
|
* also supported.
|
||||||
|
*
|
||||||
|
* PHYSICAL address
|
||||||
|
* space
|
||||||
|
*
|
||||||
|
* | |
|
||||||
|
* ...+----+--> mem_start
|
||||||
|
* VIRTUAL address ... | |
|
||||||
|
* space ... | | [PART_0]
|
||||||
|
* ... | |
|
||||||
|
* 0x00000000 <--+----+... ...+----+--> mem_start + mem_size
|
||||||
|
* | | ... | |
|
||||||
|
* |MEM | ... | |
|
||||||
|
* | | ... | |
|
||||||
|
* part_size <--+----+... | | {unused area)
|
||||||
|
* | | ... | |
|
||||||
|
* |REG | ... | |
|
||||||
|
* part_size | | ... | |
|
||||||
|
* + <--+----+... ...+----+--> reg_start
|
||||||
|
* reg_size ... | |
|
||||||
|
* ... | | [PART_1]
|
||||||
|
* ... | |
|
||||||
|
* ...+----+--> reg_start + reg_size
|
||||||
|
* | |
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
void wl1251_set_partition(struct wl1251 *wl,
|
||||||
|
u32 mem_start, u32 mem_size,
|
||||||
|
u32 reg_start, u32 reg_size)
|
||||||
|
{
|
||||||
|
struct wl1251_partition partition[2];
|
||||||
|
|
||||||
|
wl1251_debug(DEBUG_SPI, "mem_start %08X mem_size %08X",
|
||||||
|
mem_start, mem_size);
|
||||||
|
wl1251_debug(DEBUG_SPI, "reg_start %08X reg_size %08X",
|
||||||
|
reg_start, reg_size);
|
||||||
|
|
||||||
|
/* Make sure that the two partitions together don't exceed the
|
||||||
|
* address range */
|
||||||
|
if ((mem_size + reg_size) > HW_ACCESS_MEMORY_MAX_RANGE) {
|
||||||
|
wl1251_debug(DEBUG_SPI, "Total size exceeds maximum virtual"
|
||||||
|
" address range. Truncating partition[0].");
|
||||||
|
mem_size = HW_ACCESS_MEMORY_MAX_RANGE - reg_size;
|
||||||
|
wl1251_debug(DEBUG_SPI, "mem_start %08X mem_size %08X",
|
||||||
|
mem_start, mem_size);
|
||||||
|
wl1251_debug(DEBUG_SPI, "reg_start %08X reg_size %08X",
|
||||||
|
reg_start, reg_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((mem_start < reg_start) &&
|
||||||
|
((mem_start + mem_size) > reg_start)) {
|
||||||
|
/* Guarantee that the memory partition doesn't overlap the
|
||||||
|
* registers partition */
|
||||||
|
wl1251_debug(DEBUG_SPI, "End of partition[0] is "
|
||||||
|
"overlapping partition[1]. Adjusted.");
|
||||||
|
mem_size = reg_start - mem_start;
|
||||||
|
wl1251_debug(DEBUG_SPI, "mem_start %08X mem_size %08X",
|
||||||
|
mem_start, mem_size);
|
||||||
|
wl1251_debug(DEBUG_SPI, "reg_start %08X reg_size %08X",
|
||||||
|
reg_start, reg_size);
|
||||||
|
} else if ((reg_start < mem_start) &&
|
||||||
|
((reg_start + reg_size) > mem_start)) {
|
||||||
|
/* Guarantee that the register partition doesn't overlap the
|
||||||
|
* memory partition */
|
||||||
|
wl1251_debug(DEBUG_SPI, "End of partition[1] is"
|
||||||
|
" overlapping partition[0]. Adjusted.");
|
||||||
|
reg_size = mem_start - reg_start;
|
||||||
|
wl1251_debug(DEBUG_SPI, "mem_start %08X mem_size %08X",
|
||||||
|
mem_start, mem_size);
|
||||||
|
wl1251_debug(DEBUG_SPI, "reg_start %08X reg_size %08X",
|
||||||
|
reg_start, reg_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
partition[0].start = mem_start;
|
||||||
|
partition[0].size = mem_size;
|
||||||
|
partition[1].start = reg_start;
|
||||||
|
partition[1].size = reg_size;
|
||||||
|
|
||||||
|
wl->physical_mem_addr = mem_start;
|
||||||
|
wl->physical_reg_addr = reg_start;
|
||||||
|
|
||||||
|
wl->virtual_mem_addr = 0;
|
||||||
|
wl->virtual_reg_addr = mem_size;
|
||||||
|
|
||||||
|
wl->if_ops->write(wl, HW_ACCESS_PART0_SIZE_ADDR, partition,
|
||||||
|
sizeof(partition));
|
||||||
|
}
|
||||||
|
|
|
@ -23,6 +23,17 @@
|
||||||
|
|
||||||
#include "wl1251.h"
|
#include "wl1251.h"
|
||||||
|
|
||||||
|
#define HW_ACCESS_MEMORY_MAX_RANGE 0x1FFC0
|
||||||
|
|
||||||
|
#define HW_ACCESS_PART0_SIZE_ADDR 0x1FFC0
|
||||||
|
#define HW_ACCESS_PART0_START_ADDR 0x1FFC4
|
||||||
|
#define HW_ACCESS_PART1_SIZE_ADDR 0x1FFC8
|
||||||
|
#define HW_ACCESS_PART1_START_ADDR 0x1FFCC
|
||||||
|
|
||||||
|
#define HW_ACCESS_REGISTER_SIZE 4
|
||||||
|
|
||||||
|
#define HW_ACCESS_PRAM_MAX_RANGE 0x3c000
|
||||||
|
|
||||||
static inline u32 wl1251_read32(struct wl1251 *wl, int addr)
|
static inline u32 wl1251_read32(struct wl1251 *wl, int addr)
|
||||||
{
|
{
|
||||||
u32 response;
|
u32 response;
|
||||||
|
@ -46,4 +57,8 @@ void wl1251_mem_write32(struct wl1251 *wl, int addr, u32 val);
|
||||||
u32 wl1251_reg_read32(struct wl1251 *wl, int addr);
|
u32 wl1251_reg_read32(struct wl1251 *wl, int addr);
|
||||||
void wl1251_reg_write32(struct wl1251 *wl, int addr, u32 val);
|
void wl1251_reg_write32(struct wl1251 *wl, int addr, u32 val);
|
||||||
|
|
||||||
|
void wl1251_set_partition(struct wl1251 *wl,
|
||||||
|
u32 part_start, u32 part_size,
|
||||||
|
u32 reg_start, u32 reg_size);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -115,130 +115,6 @@ static void wl1251_spi_reset_wake(struct wl1251 *wl)
|
||||||
wl1251_spi_init(wl);
|
wl1251_spi_init(wl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Set the SPI partitions to access the chip addresses
|
|
||||||
*
|
|
||||||
* There are two VIRTUAL (SPI) partitions (the memory partition and the
|
|
||||||
* registers partition), which are mapped to two different areas of the
|
|
||||||
* PHYSICAL (hardware) memory. This function also makes other checks to
|
|
||||||
* ensure that the partitions are not overlapping. In the diagram below, the
|
|
||||||
* memory partition comes before the register partition, but the opposite is
|
|
||||||
* also supported.
|
|
||||||
*
|
|
||||||
* PHYSICAL address
|
|
||||||
* space
|
|
||||||
*
|
|
||||||
* | |
|
|
||||||
* ...+----+--> mem_start
|
|
||||||
* VIRTUAL address ... | |
|
|
||||||
* space ... | | [PART_0]
|
|
||||||
* ... | |
|
|
||||||
* 0x00000000 <--+----+... ...+----+--> mem_start + mem_size
|
|
||||||
* | | ... | |
|
|
||||||
* |MEM | ... | |
|
|
||||||
* | | ... | |
|
|
||||||
* part_size <--+----+... | | {unused area)
|
|
||||||
* | | ... | |
|
|
||||||
* |REG | ... | |
|
|
||||||
* part_size | | ... | |
|
|
||||||
* + <--+----+... ...+----+--> reg_start
|
|
||||||
* reg_size ... | |
|
|
||||||
* ... | | [PART_1]
|
|
||||||
* ... | |
|
|
||||||
* ...+----+--> reg_start + reg_size
|
|
||||||
* | |
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
int wl1251_set_partition(struct wl1251 *wl,
|
|
||||||
u32 mem_start, u32 mem_size,
|
|
||||||
u32 reg_start, u32 reg_size)
|
|
||||||
{
|
|
||||||
struct wl1251_partition *partition;
|
|
||||||
struct spi_transfer t;
|
|
||||||
struct spi_message m;
|
|
||||||
size_t len, cmd_len;
|
|
||||||
u32 *cmd;
|
|
||||||
int addr;
|
|
||||||
|
|
||||||
cmd_len = sizeof(u32) + 2 * sizeof(struct wl1251_partition);
|
|
||||||
cmd = kzalloc(cmd_len, GFP_KERNEL);
|
|
||||||
if (!cmd)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
spi_message_init(&m);
|
|
||||||
memset(&t, 0, sizeof(t));
|
|
||||||
|
|
||||||
partition = (struct wl1251_partition *) (cmd + 1);
|
|
||||||
addr = HW_ACCESS_PART0_SIZE_ADDR;
|
|
||||||
len = 2 * sizeof(struct wl1251_partition);
|
|
||||||
|
|
||||||
*cmd |= WSPI_CMD_WRITE;
|
|
||||||
*cmd |= (len << WSPI_CMD_BYTE_LENGTH_OFFSET) & WSPI_CMD_BYTE_LENGTH;
|
|
||||||
*cmd |= addr & WSPI_CMD_BYTE_ADDR;
|
|
||||||
|
|
||||||
wl1251_debug(DEBUG_SPI, "mem_start %08X mem_size %08X",
|
|
||||||
mem_start, mem_size);
|
|
||||||
wl1251_debug(DEBUG_SPI, "reg_start %08X reg_size %08X",
|
|
||||||
reg_start, reg_size);
|
|
||||||
|
|
||||||
/* Make sure that the two partitions together don't exceed the
|
|
||||||
* address range */
|
|
||||||
if ((mem_size + reg_size) > HW_ACCESS_MEMORY_MAX_RANGE) {
|
|
||||||
wl1251_debug(DEBUG_SPI, "Total size exceeds maximum virtual"
|
|
||||||
" address range. Truncating partition[0].");
|
|
||||||
mem_size = HW_ACCESS_MEMORY_MAX_RANGE - reg_size;
|
|
||||||
wl1251_debug(DEBUG_SPI, "mem_start %08X mem_size %08X",
|
|
||||||
mem_start, mem_size);
|
|
||||||
wl1251_debug(DEBUG_SPI, "reg_start %08X reg_size %08X",
|
|
||||||
reg_start, reg_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((mem_start < reg_start) &&
|
|
||||||
((mem_start + mem_size) > reg_start)) {
|
|
||||||
/* Guarantee that the memory partition doesn't overlap the
|
|
||||||
* registers partition */
|
|
||||||
wl1251_debug(DEBUG_SPI, "End of partition[0] is "
|
|
||||||
"overlapping partition[1]. Adjusted.");
|
|
||||||
mem_size = reg_start - mem_start;
|
|
||||||
wl1251_debug(DEBUG_SPI, "mem_start %08X mem_size %08X",
|
|
||||||
mem_start, mem_size);
|
|
||||||
wl1251_debug(DEBUG_SPI, "reg_start %08X reg_size %08X",
|
|
||||||
reg_start, reg_size);
|
|
||||||
} else if ((reg_start < mem_start) &&
|
|
||||||
((reg_start + reg_size) > mem_start)) {
|
|
||||||
/* Guarantee that the register partition doesn't overlap the
|
|
||||||
* memory partition */
|
|
||||||
wl1251_debug(DEBUG_SPI, "End of partition[1] is"
|
|
||||||
" overlapping partition[0]. Adjusted.");
|
|
||||||
reg_size = mem_start - reg_start;
|
|
||||||
wl1251_debug(DEBUG_SPI, "mem_start %08X mem_size %08X",
|
|
||||||
mem_start, mem_size);
|
|
||||||
wl1251_debug(DEBUG_SPI, "reg_start %08X reg_size %08X",
|
|
||||||
reg_start, reg_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
partition[0].start = mem_start;
|
|
||||||
partition[0].size = mem_size;
|
|
||||||
partition[1].start = reg_start;
|
|
||||||
partition[1].size = reg_size;
|
|
||||||
|
|
||||||
wl->physical_mem_addr = mem_start;
|
|
||||||
wl->physical_reg_addr = reg_start;
|
|
||||||
|
|
||||||
wl->virtual_mem_addr = 0;
|
|
||||||
wl->virtual_reg_addr = mem_size;
|
|
||||||
|
|
||||||
t.tx_buf = cmd;
|
|
||||||
t.len = cmd_len;
|
|
||||||
spi_message_add_tail(&t, &m);
|
|
||||||
|
|
||||||
spi_sync(wl->spi, &m);
|
|
||||||
|
|
||||||
kfree(cmd);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void wl1251_spi_read(struct wl1251 *wl, int addr, void *buf,
|
static void wl1251_spi_read(struct wl1251 *wl, int addr, void *buf,
|
||||||
size_t len)
|
size_t len)
|
||||||
{
|
{
|
||||||
|
|
|
@ -29,17 +29,6 @@
|
||||||
#include "wl1251_acx.h"
|
#include "wl1251_acx.h"
|
||||||
#include "reg.h"
|
#include "reg.h"
|
||||||
|
|
||||||
#define HW_ACCESS_MEMORY_MAX_RANGE 0x1FFC0
|
|
||||||
|
|
||||||
#define HW_ACCESS_PART0_SIZE_ADDR 0x1FFC0
|
|
||||||
#define HW_ACCESS_PART0_START_ADDR 0x1FFC4
|
|
||||||
#define HW_ACCESS_PART1_SIZE_ADDR 0x1FFC8
|
|
||||||
#define HW_ACCESS_PART1_START_ADDR 0x1FFCC
|
|
||||||
|
|
||||||
#define HW_ACCESS_REGISTER_SIZE 4
|
|
||||||
|
|
||||||
#define HW_ACCESS_PRAM_MAX_RANGE 0x3c000
|
|
||||||
|
|
||||||
#define WSPI_CMD_READ 0x40000000
|
#define WSPI_CMD_READ 0x40000000
|
||||||
#define WSPI_CMD_WRITE 0x00000000
|
#define WSPI_CMD_WRITE 0x00000000
|
||||||
#define WSPI_CMD_FIXED 0x20000000
|
#define WSPI_CMD_FIXED 0x20000000
|
||||||
|
@ -69,8 +58,4 @@
|
||||||
((WL1251_BUSY_WORD_LEN - 4) / sizeof(u32))
|
((WL1251_BUSY_WORD_LEN - 4) / sizeof(u32))
|
||||||
#define HW_ACCESS_WSPI_INIT_CMD_MASK 0
|
#define HW_ACCESS_WSPI_INIT_CMD_MASK 0
|
||||||
|
|
||||||
int wl1251_set_partition(struct wl1251 *wl,
|
|
||||||
u32 part_start, u32 part_size,
|
|
||||||
u32 reg_start, u32 reg_size);
|
|
||||||
|
|
||||||
#endif /* __WL1251_SPI_H__ */
|
#endif /* __WL1251_SPI_H__ */
|
||||||
|
|
Loading…
Reference in New Issue