mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-11-16 08:44:21 +08:00
26caddf274
The mmc block driver does not support parallel dispatch of requests. In
normal circumstances, all requests are anyway funneled through a single
work item, so parallel dispatch never happens. However it can happen if
there is no elevator.
Fix that by detecting if a dispatch is in progress and returning busy
(BLK_STS_RESOURCE) in that case
Fixes: 81196976ed
("mmc: block: Add blk-mq support")
Cc: stable@vger.kernel.org # v4.16+
Signed-off-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
125 lines
2.9 KiB
C
125 lines
2.9 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
#ifndef MMC_QUEUE_H
|
|
#define MMC_QUEUE_H
|
|
|
|
#include <linux/types.h>
|
|
#include <linux/blkdev.h>
|
|
#include <linux/blk-mq.h>
|
|
#include <linux/mmc/core.h>
|
|
#include <linux/mmc/host.h>
|
|
|
|
enum mmc_issued {
|
|
MMC_REQ_STARTED,
|
|
MMC_REQ_BUSY,
|
|
MMC_REQ_FAILED_TO_START,
|
|
MMC_REQ_FINISHED,
|
|
};
|
|
|
|
enum mmc_issue_type {
|
|
MMC_ISSUE_SYNC,
|
|
MMC_ISSUE_DCMD,
|
|
MMC_ISSUE_ASYNC,
|
|
MMC_ISSUE_MAX,
|
|
};
|
|
|
|
static inline struct mmc_queue_req *req_to_mmc_queue_req(struct request *rq)
|
|
{
|
|
return blk_mq_rq_to_pdu(rq);
|
|
}
|
|
|
|
struct mmc_queue_req;
|
|
|
|
static inline struct request *mmc_queue_req_to_req(struct mmc_queue_req *mqr)
|
|
{
|
|
return blk_mq_rq_from_pdu(mqr);
|
|
}
|
|
|
|
struct mmc_blk_data;
|
|
struct mmc_blk_ioc_data;
|
|
|
|
struct mmc_blk_request {
|
|
struct mmc_request mrq;
|
|
struct mmc_command sbc;
|
|
struct mmc_command cmd;
|
|
struct mmc_command stop;
|
|
struct mmc_data data;
|
|
};
|
|
|
|
/**
|
|
* enum mmc_drv_op - enumerates the operations in the mmc_queue_req
|
|
* @MMC_DRV_OP_IOCTL: ioctl operation
|
|
* @MMC_DRV_OP_IOCTL_RPMB: RPMB-oriented ioctl operation
|
|
* @MMC_DRV_OP_BOOT_WP: write protect boot partitions
|
|
* @MMC_DRV_OP_GET_CARD_STATUS: get card status
|
|
* @MMC_DRV_OP_GET_EXT_CSD: get the EXT CSD from an eMMC card
|
|
*/
|
|
enum mmc_drv_op {
|
|
MMC_DRV_OP_IOCTL,
|
|
MMC_DRV_OP_IOCTL_RPMB,
|
|
MMC_DRV_OP_BOOT_WP,
|
|
MMC_DRV_OP_GET_CARD_STATUS,
|
|
MMC_DRV_OP_GET_EXT_CSD,
|
|
};
|
|
|
|
struct mmc_queue_req {
|
|
struct mmc_blk_request brq;
|
|
struct scatterlist *sg;
|
|
enum mmc_drv_op drv_op;
|
|
int drv_op_result;
|
|
void *drv_op_data;
|
|
unsigned int ioc_count;
|
|
int retries;
|
|
};
|
|
|
|
struct mmc_queue {
|
|
struct mmc_card *card;
|
|
struct mmc_ctx ctx;
|
|
struct blk_mq_tag_set tag_set;
|
|
struct mmc_blk_data *blkdata;
|
|
struct request_queue *queue;
|
|
int in_flight[MMC_ISSUE_MAX];
|
|
unsigned int cqe_busy;
|
|
#define MMC_CQE_DCMD_BUSY BIT(0)
|
|
#define MMC_CQE_QUEUE_FULL BIT(1)
|
|
bool busy;
|
|
bool use_cqe;
|
|
bool recovery_needed;
|
|
bool in_recovery;
|
|
bool rw_wait;
|
|
bool waiting;
|
|
struct work_struct recovery_work;
|
|
wait_queue_head_t wait;
|
|
struct request *recovery_req;
|
|
struct request *complete_req;
|
|
struct mutex complete_lock;
|
|
struct work_struct complete_work;
|
|
};
|
|
|
|
extern int mmc_init_queue(struct mmc_queue *, struct mmc_card *, spinlock_t *,
|
|
const char *);
|
|
extern void mmc_cleanup_queue(struct mmc_queue *);
|
|
extern void mmc_queue_suspend(struct mmc_queue *);
|
|
extern void mmc_queue_resume(struct mmc_queue *);
|
|
extern unsigned int mmc_queue_map_sg(struct mmc_queue *,
|
|
struct mmc_queue_req *);
|
|
|
|
void mmc_cqe_check_busy(struct mmc_queue *mq);
|
|
void mmc_cqe_recovery_notifier(struct mmc_request *mrq);
|
|
|
|
enum mmc_issue_type mmc_issue_type(struct mmc_queue *mq, struct request *req);
|
|
|
|
static inline int mmc_tot_in_flight(struct mmc_queue *mq)
|
|
{
|
|
return mq->in_flight[MMC_ISSUE_SYNC] +
|
|
mq->in_flight[MMC_ISSUE_DCMD] +
|
|
mq->in_flight[MMC_ISSUE_ASYNC];
|
|
}
|
|
|
|
static inline int mmc_cqe_qcnt(struct mmc_queue *mq)
|
|
{
|
|
return mq->in_flight[MMC_ISSUE_DCMD] +
|
|
mq->in_flight[MMC_ISSUE_ASYNC];
|
|
}
|
|
|
|
#endif
|