Commit 45474475 authored by H. Nikolaus Schaller's avatar H. Nikolaus Schaller Committed by Kalle Valo

wl1251: dynamically allocate memory used for DMA

With introduction of vmap'ed stacks, stack parameters can no
longer be used for DMA and now leads to kernel panic.

It happens at several places for the wl1251 (e.g. when
accessed through SDIO) making it unuseable on e.g. the
OpenPandora.

We solve this by allocating temporary buffers or use wl1251_read32().

Tested on v5.18-rc5 with OpenPandora.

Fixes: a1c510d0 ("ARM: implement support for vmap'ed stacks")
Signed-off-by: default avatarH. Nikolaus Schaller <hns@goldelico.com>
Signed-off-by: default avatarKalle Valo <kvalo@kernel.org>
Link: https://lore.kernel.org/r/1676021ae8b6d7aada0b1806fed99b1b8359bdc4.1651495112.git.hns@goldelico.com
parent 193eb523
...@@ -169,11 +169,9 @@ int wl1251_event_wait(struct wl1251 *wl, u32 mask, int timeout_ms) ...@@ -169,11 +169,9 @@ int wl1251_event_wait(struct wl1251 *wl, u32 mask, int timeout_ms)
msleep(1); msleep(1);
/* read from both event fields */ /* read from both event fields */
wl1251_mem_read(wl, wl->mbox_ptr[0], &events_vector, events_vector = wl1251_mem_read32(wl, wl->mbox_ptr[0]);
sizeof(events_vector));
event = events_vector & mask; event = events_vector & mask;
wl1251_mem_read(wl, wl->mbox_ptr[1], &events_vector, events_vector = wl1251_mem_read32(wl, wl->mbox_ptr[1]);
sizeof(events_vector));
event |= events_vector & mask; event |= events_vector & mask;
} while (!event); } while (!event);
...@@ -202,7 +200,7 @@ void wl1251_event_mbox_config(struct wl1251 *wl) ...@@ -202,7 +200,7 @@ void wl1251_event_mbox_config(struct wl1251 *wl)
int wl1251_event_handle(struct wl1251 *wl, u8 mbox_num) int wl1251_event_handle(struct wl1251 *wl, u8 mbox_num)
{ {
struct event_mailbox mbox; struct event_mailbox *mbox;
int ret; int ret;
wl1251_debug(DEBUG_EVENT, "EVENT on mbox %d", mbox_num); wl1251_debug(DEBUG_EVENT, "EVENT on mbox %d", mbox_num);
...@@ -210,12 +208,20 @@ int wl1251_event_handle(struct wl1251 *wl, u8 mbox_num) ...@@ -210,12 +208,20 @@ int wl1251_event_handle(struct wl1251 *wl, u8 mbox_num)
if (mbox_num > 1) if (mbox_num > 1)
return -EINVAL; return -EINVAL;
mbox = kmalloc(sizeof(*mbox), GFP_KERNEL);
if (!mbox) {
wl1251_error("can not allocate mbox buffer");
return -ENOMEM;
}
/* first we read the mbox descriptor */ /* first we read the mbox descriptor */
wl1251_mem_read(wl, wl->mbox_ptr[mbox_num], &mbox, wl1251_mem_read(wl, wl->mbox_ptr[mbox_num], mbox,
sizeof(struct event_mailbox)); sizeof(*mbox));
/* process the descriptor */ /* process the descriptor */
ret = wl1251_event_process(wl, &mbox); ret = wl1251_event_process(wl, mbox);
kfree(mbox);
if (ret < 0) if (ret < 0)
return ret; return ret;
......
...@@ -121,7 +121,13 @@ void wl1251_set_partition(struct wl1251 *wl, ...@@ -121,7 +121,13 @@ void wl1251_set_partition(struct wl1251 *wl,
u32 mem_start, u32 mem_size, u32 mem_start, u32 mem_size,
u32 reg_start, u32 reg_size) u32 reg_start, u32 reg_size)
{ {
struct wl1251_partition partition[2]; struct wl1251_partition_set *partition;
partition = kmalloc(sizeof(*partition), GFP_KERNEL);
if (!partition) {
wl1251_error("can not allocate partition buffer");
return;
}
wl1251_debug(DEBUG_SPI, "mem_start %08X mem_size %08X", wl1251_debug(DEBUG_SPI, "mem_start %08X mem_size %08X",
mem_start, mem_size); mem_start, mem_size);
...@@ -164,10 +170,10 @@ void wl1251_set_partition(struct wl1251 *wl, ...@@ -164,10 +170,10 @@ void wl1251_set_partition(struct wl1251 *wl,
reg_start, reg_size); reg_start, reg_size);
} }
partition[0].start = mem_start; partition->mem.start = mem_start;
partition[0].size = mem_size; partition->mem.size = mem_size;
partition[1].start = reg_start; partition->reg.start = reg_start;
partition[1].size = reg_size; partition->reg.size = reg_size;
wl->physical_mem_addr = mem_start; wl->physical_mem_addr = mem_start;
wl->physical_reg_addr = reg_start; wl->physical_reg_addr = reg_start;
...@@ -176,5 +182,7 @@ void wl1251_set_partition(struct wl1251 *wl, ...@@ -176,5 +182,7 @@ void wl1251_set_partition(struct wl1251 *wl,
wl->virtual_reg_addr = mem_size; wl->virtual_reg_addr = mem_size;
wl->if_ops->write(wl, HW_ACCESS_PART0_SIZE_ADDR, partition, wl->if_ops->write(wl, HW_ACCESS_PART0_SIZE_ADDR, partition,
sizeof(partition)); sizeof(*partition));
kfree(partition);
} }
...@@ -443,19 +443,25 @@ static void wl1251_tx_packet_cb(struct wl1251 *wl, ...@@ -443,19 +443,25 @@ static void wl1251_tx_packet_cb(struct wl1251 *wl,
void wl1251_tx_complete(struct wl1251 *wl) void wl1251_tx_complete(struct wl1251 *wl)
{ {
int i, result_index, num_complete = 0, queue_len; int i, result_index, num_complete = 0, queue_len;
struct tx_result result[FW_TX_CMPLT_BLOCK_SIZE], *result_ptr; struct tx_result *result, *result_ptr;
unsigned long flags; unsigned long flags;
if (unlikely(wl->state != WL1251_STATE_ON)) if (unlikely(wl->state != WL1251_STATE_ON))
return; return;
result = kmalloc_array(FW_TX_CMPLT_BLOCK_SIZE, sizeof(*result), GFP_KERNEL);
if (!result) {
wl1251_error("can not allocate result buffer");
return;
}
/* First we read the result */ /* First we read the result */
wl1251_mem_read(wl, wl->data_path->tx_complete_addr, wl1251_mem_read(wl, wl->data_path->tx_complete_addr, result,
result, sizeof(result)); FW_TX_CMPLT_BLOCK_SIZE * sizeof(*result));
result_index = wl->next_tx_complete; result_index = wl->next_tx_complete;
for (i = 0; i < ARRAY_SIZE(result); i++) { for (i = 0; i < FW_TX_CMPLT_BLOCK_SIZE; i++) {
result_ptr = &result[result_index]; result_ptr = &result[result_index];
if (result_ptr->done_1 == 1 && if (result_ptr->done_1 == 1 &&
...@@ -538,6 +544,7 @@ void wl1251_tx_complete(struct wl1251 *wl) ...@@ -538,6 +544,7 @@ void wl1251_tx_complete(struct wl1251 *wl)
} }
kfree(result);
wl->next_tx_complete = result_index; wl->next_tx_complete = result_index;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment