queue.h 2.9 KB
Newer Older
1
/* SPDX-License-Identifier: GPL-2.0 */
L
Linus Torvalds 已提交
2 3 4
#ifndef MMC_QUEUE_H
#define MMC_QUEUE_H

5 6
#include <linux/types.h>
#include <linux/blkdev.h>
7
#include <linux/blk-mq.h>
8 9 10
#include <linux/mmc/core.h>
#include <linux/mmc/host.h>

A
Adrian Hunter 已提交
11 12 13 14 15 16 17 18 19
enum mmc_issued {
	MMC_REQ_STARTED,
	MMC_REQ_BUSY,
	MMC_REQ_FAILED_TO_START,
	MMC_REQ_FINISHED,
};

enum mmc_issue_type {
	MMC_ISSUE_SYNC,
A
Adrian Hunter 已提交
20
	MMC_ISSUE_DCMD,
A
Adrian Hunter 已提交
21 22 23 24
	MMC_ISSUE_ASYNC,
	MMC_ISSUE_MAX,
};

25 26 27 28 29
static inline struct mmc_queue_req *req_to_mmc_queue_req(struct request *rq)
{
	return blk_mq_rq_to_pdu(rq);
}

30 31 32 33 34 35 36
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);
}

L
Linus Walleij 已提交
37
struct mmc_blk_data;
38
struct mmc_blk_ioc_data;
L
Linus Torvalds 已提交
39

40 41 42 43 44 45 46 47
struct mmc_blk_request {
	struct mmc_request	mrq;
	struct mmc_command	sbc;
	struct mmc_command	cmd;
	struct mmc_command	stop;
	struct mmc_data		data;
};

48 49 50
/**
 * enum mmc_drv_op - enumerates the operations in the mmc_queue_req
 * @MMC_DRV_OP_IOCTL: ioctl operation
51
 * @MMC_DRV_OP_IOCTL_RPMB: RPMB-oriented ioctl operation
52
 * @MMC_DRV_OP_BOOT_WP: write protect boot partitions
53 54
 * @MMC_DRV_OP_GET_CARD_STATUS: get card status
 * @MMC_DRV_OP_GET_EXT_CSD: get the EXT CSD from an eMMC card
55 56 57
 */
enum mmc_drv_op {
	MMC_DRV_OP_IOCTL,
58
	MMC_DRV_OP_IOCTL_RPMB,
59
	MMC_DRV_OP_BOOT_WP,
60 61
	MMC_DRV_OP_GET_CARD_STATUS,
	MMC_DRV_OP_GET_EXT_CSD,
62 63
};

64 65 66
struct mmc_queue_req {
	struct mmc_blk_request	brq;
	struct scatterlist	*sg;
67
	enum mmc_drv_op		drv_op;
68
	int			drv_op_result;
69
	void			*drv_op_data;
70
	unsigned int		ioc_count;
A
Adrian Hunter 已提交
71
	int			retries;
72 73
};

L
Linus Torvalds 已提交
74 75
struct mmc_queue {
	struct mmc_card		*card;
A
Adrian Hunter 已提交
76 77
	struct mmc_ctx		ctx;
	struct blk_mq_tag_set	tag_set;
L
Linus Walleij 已提交
78
	struct mmc_blk_data	*blkdata;
L
Linus Torvalds 已提交
79
	struct request_queue	*queue;
A
Adrian Hunter 已提交
80
	int			in_flight[MMC_ISSUE_MAX];
A
Adrian Hunter 已提交
81 82 83 84 85 86
	unsigned int		cqe_busy;
#define MMC_CQE_DCMD_BUSY	BIT(0)
#define MMC_CQE_QUEUE_FULL	BIT(1)
	bool			use_cqe;
	bool			recovery_needed;
	bool			in_recovery;
A
Adrian Hunter 已提交
87 88
	bool			rw_wait;
	bool			waiting;
A
Adrian Hunter 已提交
89
	struct work_struct	recovery_work;
A
Adrian Hunter 已提交
90
	wait_queue_head_t	wait;
91
	struct request		*recovery_req;
A
Adrian Hunter 已提交
92 93 94
	struct request		*complete_req;
	struct mutex		complete_lock;
	struct work_struct	complete_work;
L
Linus Torvalds 已提交
95 96
};

97 98
extern int mmc_init_queue(struct mmc_queue *, struct mmc_card *, spinlock_t *,
			  const char *);
L
Linus Torvalds 已提交
99 100 101
extern void mmc_cleanup_queue(struct mmc_queue *);
extern void mmc_queue_suspend(struct mmc_queue *);
extern void mmc_queue_resume(struct mmc_queue *);
102 103
extern unsigned int mmc_queue_map_sg(struct mmc_queue *,
				     struct mmc_queue_req *);
104

A
Adrian Hunter 已提交
105 106 107
void mmc_cqe_check_busy(struct mmc_queue *mq);
void mmc_cqe_recovery_notifier(struct mmc_request *mrq);

A
Adrian Hunter 已提交
108 109 110 111 112
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] +
A
Adrian Hunter 已提交
113 114 115 116 117 118 119
	       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] +
A
Adrian Hunter 已提交
120 121 122
	       mq->in_flight[MMC_ISSUE_ASYNC];
}

L
Linus Torvalds 已提交
123
#endif