123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118 |
- From: Hante Meuleman <meuleman@broadcom.com>
- Date: Tue, 19 Jan 2016 12:39:24 +0100
- Subject: [PATCH] brcmfmac: fix sdio sg table alloc crash
- With commit 7d34b0560567 ("brcmfmac: Move all module parameters to
- one place") a bug was introduced causing a null pointer exception.
- This patch fixes the bug by initializing the sg table till after
- the settings have been initialized.
- Fixes: 7d34b0560567 ("brcmfmac: Move all module parameters to one place")
- Reported-by: Marc Zyngier <marc.zyngier@arm.com>
- Tested-by: Marc Zyngier <marc.zyngier@arm.com>
- Reviewed-by: Arend Van Spriel <arend@broadcom.com>
- Reviewed-by: Franky (Zhenhui) Lin <frankyl@broadcom.com>
- Reviewed-by: Pieter-Paul Giesberts <pieterpg@broadcom.com>
- Signed-off-by: Hante Meuleman <meuleman@broadcom.com>
- Signed-off-by: Arend van Spriel <arend@broadcom.com>
- Signed-off-by: Kalle Valo <kvalo@codeaurora.org>
- ---
- --- a/drivers/net/wireless/broadcom/brcm80211/brcmfmac/bcmsdh.c
- +++ b/drivers/net/wireless/broadcom/brcm80211/brcmfmac/bcmsdh.c
- @@ -879,11 +879,24 @@ int brcmf_sdiod_abort(struct brcmf_sdio_
- return 0;
- }
-
- -static void brcmf_sdiod_sgtable_alloc(struct brcmf_sdio_dev *sdiodev)
- +void brcmf_sdiod_sgtable_alloc(struct brcmf_sdio_dev *sdiodev)
- {
- + struct sdio_func *func;
- + struct mmc_host *host;
- + uint max_blocks;
- uint nents;
- int err;
-
- + func = sdiodev->func[2];
- + host = func->card->host;
- + sdiodev->sg_support = host->max_segs > 1;
- + max_blocks = min_t(uint, host->max_blk_count, 511u);
- + sdiodev->max_request_size = min_t(uint, host->max_req_size,
- + max_blocks * func->cur_blksize);
- + sdiodev->max_segment_count = min_t(uint, host->max_segs,
- + SG_MAX_SINGLE_ALLOC);
- + sdiodev->max_segment_size = host->max_seg_size;
- +
- if (!sdiodev->sg_support)
- return;
-
- @@ -1021,9 +1034,6 @@ static void brcmf_sdiod_host_fixup(struc
-
- static int brcmf_sdiod_probe(struct brcmf_sdio_dev *sdiodev)
- {
- - struct sdio_func *func;
- - struct mmc_host *host;
- - uint max_blocks;
- int ret = 0;
-
- sdiodev->num_funcs = 2;
- @@ -1054,26 +1064,6 @@ static int brcmf_sdiod_probe(struct brcm
- goto out;
- }
-
- - /*
- - * determine host related variables after brcmf_sdiod_probe()
- - * as func->cur_blksize is properly set and F2 init has been
- - * completed successfully.
- - */
- - func = sdiodev->func[2];
- - host = func->card->host;
- - sdiodev->sg_support = host->max_segs > 1;
- - max_blocks = min_t(uint, host->max_blk_count, 511u);
- - sdiodev->max_request_size = min_t(uint, host->max_req_size,
- - max_blocks * func->cur_blksize);
- - sdiodev->max_segment_count = min_t(uint, host->max_segs,
- - SG_MAX_SINGLE_ALLOC);
- - sdiodev->max_segment_size = host->max_seg_size;
- -
- - /* allocate scatter-gather table. sg support
- - * will be disabled upon allocation failure.
- - */
- - brcmf_sdiod_sgtable_alloc(sdiodev);
- -
- ret = brcmf_sdiod_freezer_attach(sdiodev);
- if (ret)
- goto out;
- @@ -1084,7 +1074,7 @@ static int brcmf_sdiod_probe(struct brcm
- ret = -ENODEV;
- goto out;
- }
- - brcmf_sdiod_host_fixup(host);
- + brcmf_sdiod_host_fixup(sdiodev->func[2]->card->host);
- out:
- if (ret)
- brcmf_sdiod_remove(sdiodev);
- --- a/drivers/net/wireless/broadcom/brcm80211/brcmfmac/sdio.c
- +++ b/drivers/net/wireless/broadcom/brcm80211/brcmfmac/sdio.c
- @@ -4114,6 +4114,11 @@ struct brcmf_sdio *brcmf_sdio_probe(stru
- goto fail;
- }
-
- + /* allocate scatter-gather table. sg support
- + * will be disabled upon allocation failure.
- + */
- + brcmf_sdiod_sgtable_alloc(bus->sdiodev);
- +
- /* Query the F2 block size, set roundup accordingly */
- bus->blocksize = bus->sdiodev->func[2]->cur_blksize;
- bus->roundup = min(max_roundup, bus->blocksize);
- --- a/drivers/net/wireless/broadcom/brcm80211/brcmfmac/sdio.h
- +++ b/drivers/net/wireless/broadcom/brcm80211/brcmfmac/sdio.h
- @@ -342,6 +342,7 @@ int brcmf_sdiod_ramrw(struct brcmf_sdio_
-
- /* Issue an abort to the specified function */
- int brcmf_sdiod_abort(struct brcmf_sdio_dev *sdiodev, uint fn);
- +void brcmf_sdiod_sgtable_alloc(struct brcmf_sdio_dev *sdiodev);
- void brcmf_sdiod_change_state(struct brcmf_sdio_dev *sdiodev,
- enum brcmf_sdiod_state state);
- #ifdef CONFIG_PM_SLEEP
|