mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 26caddf274
			
		
	
	
		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
 |