xref: /OK3568_Linux_fs/kernel/drivers/gpu/drm/nouveau/nvkm/falcon/cmdq.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * Copyright (c) 2017, NVIDIA CORPORATION. All rights reserved.
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  * Permission is hereby granted, free of charge, to any person obtaining a
5*4882a593Smuzhiyun  * copy of this software and associated documentation files (the "Software"),
6*4882a593Smuzhiyun  * to deal in the Software without restriction, including without limitation
7*4882a593Smuzhiyun  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*4882a593Smuzhiyun  * and/or sell copies of the Software, and to permit persons to whom the
9*4882a593Smuzhiyun  * Software is furnished to do so, subject to the following conditions:
10*4882a593Smuzhiyun  *
11*4882a593Smuzhiyun  * The above copyright notice and this permission notice shall be included in
12*4882a593Smuzhiyun  * all copies or substantial portions of the Software.
13*4882a593Smuzhiyun  *
14*4882a593Smuzhiyun  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15*4882a593Smuzhiyun  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16*4882a593Smuzhiyun  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
17*4882a593Smuzhiyun  * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18*4882a593Smuzhiyun  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19*4882a593Smuzhiyun  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20*4882a593Smuzhiyun  * OTHER DEALINGS IN THE SOFTWARE.
21*4882a593Smuzhiyun  *
22*4882a593Smuzhiyun  */
23*4882a593Smuzhiyun #include "qmgr.h"
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun static bool
nvkm_falcon_cmdq_has_room(struct nvkm_falcon_cmdq * cmdq,u32 size,bool * rewind)26*4882a593Smuzhiyun nvkm_falcon_cmdq_has_room(struct nvkm_falcon_cmdq *cmdq, u32 size, bool *rewind)
27*4882a593Smuzhiyun {
28*4882a593Smuzhiyun 	u32 head = nvkm_falcon_rd32(cmdq->qmgr->falcon, cmdq->head_reg);
29*4882a593Smuzhiyun 	u32 tail = nvkm_falcon_rd32(cmdq->qmgr->falcon, cmdq->tail_reg);
30*4882a593Smuzhiyun 	u32 free;
31*4882a593Smuzhiyun 
32*4882a593Smuzhiyun 	size = ALIGN(size, QUEUE_ALIGNMENT);
33*4882a593Smuzhiyun 
34*4882a593Smuzhiyun 	if (head >= tail) {
35*4882a593Smuzhiyun 		free = cmdq->offset + cmdq->size - head;
36*4882a593Smuzhiyun 		free -= HDR_SIZE;
37*4882a593Smuzhiyun 
38*4882a593Smuzhiyun 		if (size > free) {
39*4882a593Smuzhiyun 			*rewind = true;
40*4882a593Smuzhiyun 			head = cmdq->offset;
41*4882a593Smuzhiyun 		}
42*4882a593Smuzhiyun 	}
43*4882a593Smuzhiyun 
44*4882a593Smuzhiyun 	if (head < tail)
45*4882a593Smuzhiyun 		free = tail - head - 1;
46*4882a593Smuzhiyun 
47*4882a593Smuzhiyun 	return size <= free;
48*4882a593Smuzhiyun }
49*4882a593Smuzhiyun 
50*4882a593Smuzhiyun static void
nvkm_falcon_cmdq_push(struct nvkm_falcon_cmdq * cmdq,void * data,u32 size)51*4882a593Smuzhiyun nvkm_falcon_cmdq_push(struct nvkm_falcon_cmdq *cmdq, void *data, u32 size)
52*4882a593Smuzhiyun {
53*4882a593Smuzhiyun 	struct nvkm_falcon *falcon = cmdq->qmgr->falcon;
54*4882a593Smuzhiyun 	nvkm_falcon_load_dmem(falcon, data, cmdq->position, size, 0);
55*4882a593Smuzhiyun 	cmdq->position += ALIGN(size, QUEUE_ALIGNMENT);
56*4882a593Smuzhiyun }
57*4882a593Smuzhiyun 
58*4882a593Smuzhiyun static void
nvkm_falcon_cmdq_rewind(struct nvkm_falcon_cmdq * cmdq)59*4882a593Smuzhiyun nvkm_falcon_cmdq_rewind(struct nvkm_falcon_cmdq *cmdq)
60*4882a593Smuzhiyun {
61*4882a593Smuzhiyun 	struct nvfw_falcon_cmd cmd;
62*4882a593Smuzhiyun 
63*4882a593Smuzhiyun 	cmd.unit_id = NV_FALCON_CMD_UNIT_ID_REWIND;
64*4882a593Smuzhiyun 	cmd.size = sizeof(cmd);
65*4882a593Smuzhiyun 	nvkm_falcon_cmdq_push(cmdq, &cmd, cmd.size);
66*4882a593Smuzhiyun 
67*4882a593Smuzhiyun 	cmdq->position = cmdq->offset;
68*4882a593Smuzhiyun }
69*4882a593Smuzhiyun 
70*4882a593Smuzhiyun static int
nvkm_falcon_cmdq_open(struct nvkm_falcon_cmdq * cmdq,u32 size)71*4882a593Smuzhiyun nvkm_falcon_cmdq_open(struct nvkm_falcon_cmdq *cmdq, u32 size)
72*4882a593Smuzhiyun {
73*4882a593Smuzhiyun 	struct nvkm_falcon *falcon = cmdq->qmgr->falcon;
74*4882a593Smuzhiyun 	bool rewind = false;
75*4882a593Smuzhiyun 
76*4882a593Smuzhiyun 	mutex_lock(&cmdq->mutex);
77*4882a593Smuzhiyun 
78*4882a593Smuzhiyun 	if (!nvkm_falcon_cmdq_has_room(cmdq, size, &rewind)) {
79*4882a593Smuzhiyun 		FLCNQ_DBG(cmdq, "queue full");
80*4882a593Smuzhiyun 		mutex_unlock(&cmdq->mutex);
81*4882a593Smuzhiyun 		return -EAGAIN;
82*4882a593Smuzhiyun 	}
83*4882a593Smuzhiyun 
84*4882a593Smuzhiyun 	cmdq->position = nvkm_falcon_rd32(falcon, cmdq->head_reg);
85*4882a593Smuzhiyun 
86*4882a593Smuzhiyun 	if (rewind)
87*4882a593Smuzhiyun 		nvkm_falcon_cmdq_rewind(cmdq);
88*4882a593Smuzhiyun 
89*4882a593Smuzhiyun 	return 0;
90*4882a593Smuzhiyun }
91*4882a593Smuzhiyun 
92*4882a593Smuzhiyun static void
nvkm_falcon_cmdq_close(struct nvkm_falcon_cmdq * cmdq)93*4882a593Smuzhiyun nvkm_falcon_cmdq_close(struct nvkm_falcon_cmdq *cmdq)
94*4882a593Smuzhiyun {
95*4882a593Smuzhiyun 	nvkm_falcon_wr32(cmdq->qmgr->falcon, cmdq->head_reg, cmdq->position);
96*4882a593Smuzhiyun 	mutex_unlock(&cmdq->mutex);
97*4882a593Smuzhiyun }
98*4882a593Smuzhiyun 
99*4882a593Smuzhiyun static int
nvkm_falcon_cmdq_write(struct nvkm_falcon_cmdq * cmdq,struct nvfw_falcon_cmd * cmd)100*4882a593Smuzhiyun nvkm_falcon_cmdq_write(struct nvkm_falcon_cmdq *cmdq, struct nvfw_falcon_cmd *cmd)
101*4882a593Smuzhiyun {
102*4882a593Smuzhiyun 	static unsigned timeout = 2000;
103*4882a593Smuzhiyun 	unsigned long end_jiffies = jiffies + msecs_to_jiffies(timeout);
104*4882a593Smuzhiyun 	int ret = -EAGAIN;
105*4882a593Smuzhiyun 
106*4882a593Smuzhiyun 	while (ret == -EAGAIN && time_before(jiffies, end_jiffies))
107*4882a593Smuzhiyun 		ret = nvkm_falcon_cmdq_open(cmdq, cmd->size);
108*4882a593Smuzhiyun 	if (ret) {
109*4882a593Smuzhiyun 		FLCNQ_ERR(cmdq, "timeout waiting for queue space");
110*4882a593Smuzhiyun 		return ret;
111*4882a593Smuzhiyun 	}
112*4882a593Smuzhiyun 
113*4882a593Smuzhiyun 	nvkm_falcon_cmdq_push(cmdq, cmd, cmd->size);
114*4882a593Smuzhiyun 	nvkm_falcon_cmdq_close(cmdq);
115*4882a593Smuzhiyun 	return ret;
116*4882a593Smuzhiyun }
117*4882a593Smuzhiyun 
118*4882a593Smuzhiyun /* specifies that we want to know the command status in the answer message */
119*4882a593Smuzhiyun #define CMD_FLAGS_STATUS BIT(0)
120*4882a593Smuzhiyun /* specifies that we want an interrupt when the answer message is queued */
121*4882a593Smuzhiyun #define CMD_FLAGS_INTR BIT(1)
122*4882a593Smuzhiyun 
123*4882a593Smuzhiyun int
nvkm_falcon_cmdq_send(struct nvkm_falcon_cmdq * cmdq,struct nvfw_falcon_cmd * cmd,nvkm_falcon_qmgr_callback cb,void * priv,unsigned long timeout)124*4882a593Smuzhiyun nvkm_falcon_cmdq_send(struct nvkm_falcon_cmdq *cmdq, struct nvfw_falcon_cmd *cmd,
125*4882a593Smuzhiyun 		      nvkm_falcon_qmgr_callback cb, void *priv,
126*4882a593Smuzhiyun 		      unsigned long timeout)
127*4882a593Smuzhiyun {
128*4882a593Smuzhiyun 	struct nvkm_falcon_qmgr_seq *seq;
129*4882a593Smuzhiyun 	int ret;
130*4882a593Smuzhiyun 
131*4882a593Smuzhiyun 	if (!wait_for_completion_timeout(&cmdq->ready,
132*4882a593Smuzhiyun 					 msecs_to_jiffies(1000))) {
133*4882a593Smuzhiyun 		FLCNQ_ERR(cmdq, "timeout waiting for queue ready");
134*4882a593Smuzhiyun 		return -ETIMEDOUT;
135*4882a593Smuzhiyun 	}
136*4882a593Smuzhiyun 
137*4882a593Smuzhiyun 	seq = nvkm_falcon_qmgr_seq_acquire(cmdq->qmgr);
138*4882a593Smuzhiyun 	if (IS_ERR(seq))
139*4882a593Smuzhiyun 		return PTR_ERR(seq);
140*4882a593Smuzhiyun 
141*4882a593Smuzhiyun 	cmd->seq_id = seq->id;
142*4882a593Smuzhiyun 	cmd->ctrl_flags = CMD_FLAGS_STATUS | CMD_FLAGS_INTR;
143*4882a593Smuzhiyun 
144*4882a593Smuzhiyun 	seq->state = SEQ_STATE_USED;
145*4882a593Smuzhiyun 	seq->async = !timeout;
146*4882a593Smuzhiyun 	seq->callback = cb;
147*4882a593Smuzhiyun 	seq->priv = priv;
148*4882a593Smuzhiyun 
149*4882a593Smuzhiyun 	ret = nvkm_falcon_cmdq_write(cmdq, cmd);
150*4882a593Smuzhiyun 	if (ret) {
151*4882a593Smuzhiyun 		seq->state = SEQ_STATE_PENDING;
152*4882a593Smuzhiyun 		nvkm_falcon_qmgr_seq_release(cmdq->qmgr, seq);
153*4882a593Smuzhiyun 		return ret;
154*4882a593Smuzhiyun 	}
155*4882a593Smuzhiyun 
156*4882a593Smuzhiyun 	if (!seq->async) {
157*4882a593Smuzhiyun 		if (!wait_for_completion_timeout(&seq->done, timeout)) {
158*4882a593Smuzhiyun 			FLCNQ_ERR(cmdq, "timeout waiting for reply");
159*4882a593Smuzhiyun 			return -ETIMEDOUT;
160*4882a593Smuzhiyun 		}
161*4882a593Smuzhiyun 		ret = seq->result;
162*4882a593Smuzhiyun 		nvkm_falcon_qmgr_seq_release(cmdq->qmgr, seq);
163*4882a593Smuzhiyun 	}
164*4882a593Smuzhiyun 
165*4882a593Smuzhiyun 	return ret;
166*4882a593Smuzhiyun }
167*4882a593Smuzhiyun 
168*4882a593Smuzhiyun void
nvkm_falcon_cmdq_fini(struct nvkm_falcon_cmdq * cmdq)169*4882a593Smuzhiyun nvkm_falcon_cmdq_fini(struct nvkm_falcon_cmdq *cmdq)
170*4882a593Smuzhiyun {
171*4882a593Smuzhiyun 	reinit_completion(&cmdq->ready);
172*4882a593Smuzhiyun }
173*4882a593Smuzhiyun 
174*4882a593Smuzhiyun void
nvkm_falcon_cmdq_init(struct nvkm_falcon_cmdq * cmdq,u32 index,u32 offset,u32 size)175*4882a593Smuzhiyun nvkm_falcon_cmdq_init(struct nvkm_falcon_cmdq *cmdq,
176*4882a593Smuzhiyun 		      u32 index, u32 offset, u32 size)
177*4882a593Smuzhiyun {
178*4882a593Smuzhiyun 	const struct nvkm_falcon_func *func = cmdq->qmgr->falcon->func;
179*4882a593Smuzhiyun 
180*4882a593Smuzhiyun 	cmdq->head_reg = func->cmdq.head + index * func->cmdq.stride;
181*4882a593Smuzhiyun 	cmdq->tail_reg = func->cmdq.tail + index * func->cmdq.stride;
182*4882a593Smuzhiyun 	cmdq->offset = offset;
183*4882a593Smuzhiyun 	cmdq->size = size;
184*4882a593Smuzhiyun 	complete_all(&cmdq->ready);
185*4882a593Smuzhiyun 
186*4882a593Smuzhiyun 	FLCNQ_DBG(cmdq, "initialised @ index %d offset 0x%08x size 0x%08x",
187*4882a593Smuzhiyun 		  index, cmdq->offset, cmdq->size);
188*4882a593Smuzhiyun }
189*4882a593Smuzhiyun 
190*4882a593Smuzhiyun void
nvkm_falcon_cmdq_del(struct nvkm_falcon_cmdq ** pcmdq)191*4882a593Smuzhiyun nvkm_falcon_cmdq_del(struct nvkm_falcon_cmdq **pcmdq)
192*4882a593Smuzhiyun {
193*4882a593Smuzhiyun 	struct nvkm_falcon_cmdq *cmdq = *pcmdq;
194*4882a593Smuzhiyun 	if (cmdq) {
195*4882a593Smuzhiyun 		kfree(*pcmdq);
196*4882a593Smuzhiyun 		*pcmdq = NULL;
197*4882a593Smuzhiyun 	}
198*4882a593Smuzhiyun }
199*4882a593Smuzhiyun 
200*4882a593Smuzhiyun int
nvkm_falcon_cmdq_new(struct nvkm_falcon_qmgr * qmgr,const char * name,struct nvkm_falcon_cmdq ** pcmdq)201*4882a593Smuzhiyun nvkm_falcon_cmdq_new(struct nvkm_falcon_qmgr *qmgr, const char *name,
202*4882a593Smuzhiyun 		     struct nvkm_falcon_cmdq **pcmdq)
203*4882a593Smuzhiyun {
204*4882a593Smuzhiyun 	struct nvkm_falcon_cmdq *cmdq = *pcmdq;
205*4882a593Smuzhiyun 
206*4882a593Smuzhiyun 	if (!(cmdq = *pcmdq = kzalloc(sizeof(*cmdq), GFP_KERNEL)))
207*4882a593Smuzhiyun 		return -ENOMEM;
208*4882a593Smuzhiyun 
209*4882a593Smuzhiyun 	cmdq->qmgr = qmgr;
210*4882a593Smuzhiyun 	cmdq->name = name;
211*4882a593Smuzhiyun 	mutex_init(&cmdq->mutex);
212*4882a593Smuzhiyun 	init_completion(&cmdq->ready);
213*4882a593Smuzhiyun 	return 0;
214*4882a593Smuzhiyun }
215