blob: eb817132bc5f980b98eba67069215fc8f3485a47 [file] [log] [blame]
Thomas Gleixner9c92ab62019-05-29 07:17:56 -07001// SPDX-License-Identifier: GPL-2.0-only
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +02002/*
3 * Remote processor messaging transport (OMAP platform-specific bits)
4 *
5 * Copyright (C) 2011 Texas Instruments, Inc.
6 * Copyright (C) 2011 Google, Inc.
7 *
8 * Ohad Ben-Cohen <ohad@wizery.com>
9 * Brian Swetland <swetland@google.com>
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020010 */
11
Loic Pallardy086d0872019-01-10 14:50:49 +010012#include <linux/dma-mapping.h>
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020013#include <linux/export.h>
Loic Pallardy086d0872019-01-10 14:50:49 +010014#include <linux/of_reserved_mem.h>
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020015#include <linux/remoteproc.h>
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020016#include <linux/virtio.h>
17#include <linux/virtio_config.h>
18#include <linux/virtio_ids.h>
19#include <linux/virtio_ring.h>
20#include <linux/err.h>
21#include <linux/kref.h>
22#include <linux/slab.h>
23
24#include "remoteproc_internal.h"
25
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020026/* kick the remote processor, and let it know which virtqueue to poke at */
Heinz Graalfs46f9c2b2013-10-29 09:38:50 +103027static bool rproc_virtio_notify(struct virtqueue *vq)
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020028{
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010029 struct rproc_vring *rvring = vq->priv;
30 struct rproc *rproc = rvring->rvdev->rproc;
31 int notifyid = rvring->notifyid;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020032
Ohad Ben-Cohenb5ab5e22012-05-30 22:01:25 +030033 dev_dbg(&rproc->dev, "kicking vq index: %d\n", notifyid);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020034
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010035 rproc->ops->kick(rproc, notifyid);
Heinz Graalfs46f9c2b2013-10-29 09:38:50 +103036 return true;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020037}
38
39/**
40 * rproc_vq_interrupt() - tell remoteproc that a virtqueue is interrupted
41 * @rproc: handle to the remote processor
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010042 * @notifyid: index of the signalled virtqueue (unique per this @rproc)
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020043 *
44 * This function should be called by the platform-specific rproc driver,
45 * when the remote processor signals that a specific virtqueue has pending
46 * messages available.
47 *
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010048 * Returns IRQ_NONE if no message was found in the @notifyid virtqueue,
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020049 * and otherwise returns IRQ_HANDLED.
50 */
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010051irqreturn_t rproc_vq_interrupt(struct rproc *rproc, int notifyid)
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020052{
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010053 struct rproc_vring *rvring;
54
Ohad Ben-Cohenb5ab5e22012-05-30 22:01:25 +030055 dev_dbg(&rproc->dev, "vq index %d is interrupted\n", notifyid);
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010056
57 rvring = idr_find(&rproc->notifyids, notifyid);
58 if (!rvring || !rvring->vq)
59 return IRQ_NONE;
60
61 return vring_interrupt(0, rvring->vq);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020062}
63EXPORT_SYMBOL(rproc_vq_interrupt);
64
65static struct virtqueue *rp_find_vq(struct virtio_device *vdev,
Anna, Sumanf1459282016-08-12 18:42:19 -050066 unsigned int id,
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020067 void (*callback)(struct virtqueue *vq),
Michael S. Tsirkinf94682d2017-03-06 18:32:29 +020068 const char *name, bool ctx)
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020069{
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010070 struct rproc_vdev *rvdev = vdev_to_rvdev(vdev);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020071 struct rproc *rproc = vdev_to_rproc(vdev);
Ohad Ben-Cohenb5ab5e22012-05-30 22:01:25 +030072 struct device *dev = &rproc->dev;
Loic Pallardyc6aed2382018-07-27 15:14:47 +020073 struct rproc_mem_entry *mem;
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010074 struct rproc_vring *rvring;
Loic Pallardyc6aed2382018-07-27 15:14:47 +020075 struct fw_rsc_vdev *rsc;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020076 struct virtqueue *vq;
77 void *addr;
Bjorn Anderssona863af52016-10-19 19:40:07 -070078 int len, size;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020079
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010080 /* we're temporarily limited to two virtqueues per rvdev */
81 if (id >= ARRAY_SIZE(rvdev->vring))
82 return ERR_PTR(-EINVAL);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020083
Michael S. Tsirkin6457f122012-09-05 21:47:45 +030084 if (!name)
85 return NULL;
86
Loic Pallardyc6aed2382018-07-27 15:14:47 +020087 /* Search allocated memory region by name */
88 mem = rproc_find_carveout_by_name(rproc, "vdev%dvring%d", rvdev->index,
89 id);
90 if (!mem || !mem->va)
91 return ERR_PTR(-ENOMEM);
92
Ohad Ben-Cohen6db20ea2012-05-17 14:23:59 +030093 rvring = &rvdev->vring[id];
Loic Pallardyc6aed2382018-07-27 15:14:47 +020094 addr = mem->va;
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010095 len = rvring->len;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +020096
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +010097 /* zero vring */
98 size = vring_size(len, rvring->align);
99 memset(addr, 0, size);
100
Loic Pallardy276ec992018-07-06 14:38:27 +0200101 dev_dbg(dev, "vring%d: va %pK qsz %d notifyid %d\n",
Anna, Suman730f84c2016-08-12 18:42:20 -0500102 id, addr, len, rvring->notifyid);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200103
Ohad Ben-Cohendd6da1c2012-01-12 09:26:57 +0200104 /*
105 * Create the new vq, and tell virtio we're not interested in
106 * the 'weak' smp barriers, since we're talking with a real device.
107 */
Michael S. Tsirkinf94682d2017-03-06 18:32:29 +0200108 vq = vring_new_virtqueue(id, len, rvring->align, vdev, false, ctx,
109 addr, rproc_virtio_notify, callback, name);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200110 if (!vq) {
Ohad Ben-Cohenb5ab5e22012-05-30 22:01:25 +0300111 dev_err(dev, "vring_new_virtqueue %s failed\n", name);
Ohad Ben-Cohen6db20ea2012-05-17 14:23:59 +0300112 rproc_free_vring(rvring);
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100113 return ERR_PTR(-ENOMEM);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200114 }
115
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100116 rvring->vq = vq;
117 vq->priv = rvring;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200118
Loic Pallardyc6aed2382018-07-27 15:14:47 +0200119 /* Update vring in resource table */
120 rsc = (void *)rproc->table_ptr + rvdev->rsc_offset;
121 rsc->vring[id].da = mem->da;
122
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200123 return vq;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200124}
125
Ohad Ben-Cohendab55bb2012-11-12 11:13:51 +0200126static void __rproc_virtio_del_vqs(struct virtio_device *vdev)
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200127{
128 struct virtqueue *vq, *n;
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100129 struct rproc_vring *rvring;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200130
131 list_for_each_entry_safe(vq, n, &vdev->vqs, list) {
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100132 rvring = vq->priv;
133 rvring->vq = NULL;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200134 vring_del_virtqueue(vq);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200135 }
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200136}
137
Ohad Ben-Cohendab55bb2012-11-12 11:13:51 +0200138static void rproc_virtio_del_vqs(struct virtio_device *vdev)
139{
Ohad Ben-Cohendab55bb2012-11-12 11:13:51 +0200140 __rproc_virtio_del_vqs(vdev);
141}
142
Anna, Sumanf1459282016-08-12 18:42:19 -0500143static int rproc_virtio_find_vqs(struct virtio_device *vdev, unsigned int nvqs,
Anna, Suman730f84c2016-08-12 18:42:20 -0500144 struct virtqueue *vqs[],
145 vq_callback_t *callbacks[],
Christoph Hellwigfb5e31d2017-02-05 18:15:22 +0100146 const char * const names[],
Michael S. Tsirkinf94682d2017-03-06 18:32:29 +0200147 const bool * ctx,
Christoph Hellwigfb5e31d2017-02-05 18:15:22 +0100148 struct irq_affinity *desc)
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200149{
Wei Wanga2299892018-12-28 10:26:26 +0800150 int i, ret, queue_idx = 0;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200151
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200152 for (i = 0; i < nvqs; ++i) {
Wei Wanga2299892018-12-28 10:26:26 +0800153 if (!names[i]) {
154 vqs[i] = NULL;
155 continue;
156 }
157
158 vqs[i] = rp_find_vq(vdev, queue_idx++, callbacks[i], names[i],
Michael S. Tsirkinf94682d2017-03-06 18:32:29 +0200159 ctx ? ctx[i] : false);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200160 if (IS_ERR(vqs[i])) {
161 ret = PTR_ERR(vqs[i]);
162 goto error;
163 }
164 }
165
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200166 return 0;
167
168error:
Ohad Ben-Cohendab55bb2012-11-12 11:13:51 +0200169 __rproc_virtio_del_vqs(vdev);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200170 return ret;
171}
172
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200173static u8 rproc_virtio_get_status(struct virtio_device *vdev)
174{
Sjur Brændeland92b38f82013-02-21 18:15:39 +0100175 struct rproc_vdev *rvdev = vdev_to_rvdev(vdev);
176 struct fw_rsc_vdev *rsc;
177
178 rsc = (void *)rvdev->rproc->table_ptr + rvdev->rsc_offset;
179
180 return rsc->status;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200181}
182
183static void rproc_virtio_set_status(struct virtio_device *vdev, u8 status)
184{
Sjur Brændeland92b38f82013-02-21 18:15:39 +0100185 struct rproc_vdev *rvdev = vdev_to_rvdev(vdev);
186 struct fw_rsc_vdev *rsc;
187
188 rsc = (void *)rvdev->rproc->table_ptr + rvdev->rsc_offset;
189
190 rsc->status = status;
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100191 dev_dbg(&vdev->dev, "status: %d\n", status);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200192}
193
194static void rproc_virtio_reset(struct virtio_device *vdev)
195{
Sjur Brændeland92b38f82013-02-21 18:15:39 +0100196 struct rproc_vdev *rvdev = vdev_to_rvdev(vdev);
197 struct fw_rsc_vdev *rsc;
198
199 rsc = (void *)rvdev->rproc->table_ptr + rvdev->rsc_offset;
200
201 rsc->status = 0;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200202 dev_dbg(&vdev->dev, "reset !\n");
203}
204
205/* provide the vdev features as retrieved from the firmware */
Michael S. Tsirkind0254772014-10-07 16:39:43 +0200206static u64 rproc_virtio_get_features(struct virtio_device *vdev)
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200207{
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100208 struct rproc_vdev *rvdev = vdev_to_rvdev(vdev);
Sjur Brændeland92b38f82013-02-21 18:15:39 +0100209 struct fw_rsc_vdev *rsc;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200210
Sjur Brændeland92b38f82013-02-21 18:15:39 +0100211 rsc = (void *)rvdev->rproc->table_ptr + rvdev->rsc_offset;
212
213 return rsc->dfeatures;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200214}
215
Tiwei Bie3a814fd2018-11-21 18:03:29 +0800216static void rproc_transport_features(struct virtio_device *vdev)
217{
218 /*
219 * Packed ring isn't enabled on remoteproc for now,
220 * because remoteproc uses vring_new_virtqueue() which
221 * creates virtio rings on preallocated memory.
222 */
223 __virtio_clear_bit(vdev, VIRTIO_F_RING_PACKED);
224}
225
Michael S. Tsirkin5c609a52014-12-04 20:20:27 +0200226static int rproc_virtio_finalize_features(struct virtio_device *vdev)
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200227{
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100228 struct rproc_vdev *rvdev = vdev_to_rvdev(vdev);
Sjur Brændeland92b38f82013-02-21 18:15:39 +0100229 struct fw_rsc_vdev *rsc;
230
231 rsc = (void *)rvdev->rproc->table_ptr + rvdev->rsc_offset;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200232
233 /* Give virtio_ring a chance to accept features */
234 vring_transport_features(vdev);
235
Tiwei Bie3a814fd2018-11-21 18:03:29 +0800236 /* Give virtio_rproc a chance to accept features. */
237 rproc_transport_features(vdev);
238
Michael S. Tsirkin93d389f2014-11-27 13:45:58 +0200239 /* Make sure we don't have any features > 32 bits! */
240 BUG_ON((u32)vdev->features != vdev->features);
241
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200242 /*
243 * Remember the finalized features of our vdev, and provide it
244 * to the remote processor once it is powered on.
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200245 */
Michael S. Tsirkine16e12b2014-10-07 16:39:42 +0200246 rsc->gfeatures = vdev->features;
Michael S. Tsirkin5c609a52014-12-04 20:20:27 +0200247
248 return 0;
Sjur Brændeland92b38f82013-02-21 18:15:39 +0100249}
250
Anna, Sumanf1459282016-08-12 18:42:19 -0500251static void rproc_virtio_get(struct virtio_device *vdev, unsigned int offset,
252 void *buf, unsigned int len)
Sjur Brændeland92b38f82013-02-21 18:15:39 +0100253{
254 struct rproc_vdev *rvdev = vdev_to_rvdev(vdev);
255 struct fw_rsc_vdev *rsc;
256 void *cfg;
257
258 rsc = (void *)rvdev->rproc->table_ptr + rvdev->rsc_offset;
259 cfg = &rsc->vring[rsc->num_of_vrings];
260
261 if (offset + len > rsc->config_len || offset + len < len) {
262 dev_err(&vdev->dev, "rproc_virtio_get: access out of bounds\n");
263 return;
264 }
265
266 memcpy(buf, cfg + offset, len);
267}
268
Anna, Sumanf1459282016-08-12 18:42:19 -0500269static void rproc_virtio_set(struct virtio_device *vdev, unsigned int offset,
270 const void *buf, unsigned int len)
Sjur Brændeland92b38f82013-02-21 18:15:39 +0100271{
272 struct rproc_vdev *rvdev = vdev_to_rvdev(vdev);
273 struct fw_rsc_vdev *rsc;
274 void *cfg;
275
276 rsc = (void *)rvdev->rproc->table_ptr + rvdev->rsc_offset;
277 cfg = &rsc->vring[rsc->num_of_vrings];
278
279 if (offset + len > rsc->config_len || offset + len < len) {
280 dev_err(&vdev->dev, "rproc_virtio_set: access out of bounds\n");
281 return;
282 }
283
284 memcpy(cfg + offset, buf, len);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200285}
286
Stephen Hemminger93503932013-02-10 15:57:38 +1030287static const struct virtio_config_ops rproc_virtio_config_ops = {
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200288 .get_features = rproc_virtio_get_features,
289 .finalize_features = rproc_virtio_finalize_features,
290 .find_vqs = rproc_virtio_find_vqs,
291 .del_vqs = rproc_virtio_del_vqs,
292 .reset = rproc_virtio_reset,
293 .set_status = rproc_virtio_set_status,
294 .get_status = rproc_virtio_get_status,
Sjur Brændeland92b38f82013-02-21 18:15:39 +0100295 .get = rproc_virtio_get,
296 .set = rproc_virtio_set,
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200297};
298
299/*
300 * This function is called whenever vdev is released, and is responsible
Ohad Ben-Cohen7183a2a2012-05-30 22:02:24 +0300301 * to decrement the remote processor's refcount which was taken when vdev was
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200302 * added.
303 *
304 * Never call this function directly; it will be called by the driver
305 * core when needed.
306 */
Bjorn Anderssonaab8d802016-10-19 19:40:06 -0700307static void rproc_virtio_dev_release(struct device *dev)
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200308{
309 struct virtio_device *vdev = dev_to_virtio(dev);
Ohad Ben-Cohen6db20ea2012-05-17 14:23:59 +0300310 struct rproc_vdev *rvdev = vdev_to_rvdev(vdev);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200311 struct rproc *rproc = vdev_to_rproc(vdev);
312
Loic Pallardyd4c036f2019-01-21 14:55:15 +0100313 kfree(vdev);
314
Bjorn Anderssonaab8d802016-10-19 19:40:06 -0700315 kref_put(&rvdev->refcount, rproc_vdev_release);
Ohad Ben-Cohen6db20ea2012-05-17 14:23:59 +0300316
Ohad Ben-Cohen7183a2a2012-05-30 22:02:24 +0300317 put_device(&rproc->dev);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200318}
319
320/**
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100321 * rproc_add_virtio_dev() - register an rproc-induced virtio device
322 * @rvdev: the remote vdev
Arnaud Pouliquen2e7d4c22020-02-12 17:19:56 +0100323 * @id: the device type identification (used to match it with a driver).
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200324 *
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100325 * This function registers a virtio device. This vdev's partent is
326 * the rproc device.
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200327 *
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100328 * Returns 0 on success or an appropriate error value otherwise.
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200329 */
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100330int rproc_add_virtio_dev(struct rproc_vdev *rvdev, int id)
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200331{
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100332 struct rproc *rproc = rvdev->rproc;
Loic Pallardy086d0872019-01-10 14:50:49 +0100333 struct device *dev = &rvdev->dev;
Loic Pallardyd4c036f2019-01-21 14:55:15 +0100334 struct virtio_device *vdev;
Loic Pallardy086d0872019-01-10 14:50:49 +0100335 struct rproc_mem_entry *mem;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200336 int ret;
337
Loic Pallardy086d0872019-01-10 14:50:49 +0100338 /* Try to find dedicated vdev buffer carveout */
339 mem = rproc_find_carveout_by_name(rproc, "vdev%dbuffer", rvdev->index);
340 if (mem) {
341 phys_addr_t pa;
342
343 if (mem->of_resm_idx != -1) {
344 struct device_node *np = rproc->dev.parent->of_node;
345
346 /* Associate reserved memory to vdev device */
347 ret = of_reserved_mem_device_init_by_idx(dev, np,
348 mem->of_resm_idx);
349 if (ret) {
350 dev_err(dev, "Can't associate reserved memory\n");
351 goto out;
352 }
353 } else {
354 if (mem->va) {
355 dev_warn(dev, "vdev %d buffer already mapped\n",
356 rvdev->index);
357 pa = rproc_va_to_pa(mem->va);
358 } else {
359 /* Use dma address as carveout no memmapped yet */
360 pa = (phys_addr_t)mem->dma;
361 }
362
363 /* Associate vdev buffer memory pool to vdev subdev */
364 ret = dma_declare_coherent_memory(dev, pa,
365 mem->da,
Stephen Rothwelld664ce72019-02-22 16:14:45 +1100366 mem->len);
Loic Pallardy086d0872019-01-10 14:50:49 +0100367 if (ret < 0) {
368 dev_err(dev, "Failed to associate buffer\n");
369 goto out;
370 }
371 }
372 }
373
Loic Pallardyd4c036f2019-01-21 14:55:15 +0100374 /* Allocate virtio device */
375 vdev = kzalloc(sizeof(*vdev), GFP_KERNEL);
376 if (!vdev) {
377 ret = -ENOMEM;
378 goto out;
379 }
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100380 vdev->id.device = id,
381 vdev->config = &rproc_virtio_config_ops,
382 vdev->dev.parent = dev;
Bjorn Anderssonaab8d802016-10-19 19:40:06 -0700383 vdev->dev.release = rproc_virtio_dev_release;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200384
385 /*
386 * We're indirectly making a non-temporary copy of the rproc pointer
387 * here, because drivers probed with this vdev will indirectly
388 * access the wrapping rproc.
389 *
390 * Therefore we must increment the rproc refcount here, and decrement
391 * it _only_ when the vdev is released.
392 */
Ohad Ben-Cohen7183a2a2012-05-30 22:02:24 +0300393 get_device(&rproc->dev);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200394
Bjorn Anderssonaab8d802016-10-19 19:40:06 -0700395 /* Reference the vdev and vring allocations */
396 kref_get(&rvdev->refcount);
397
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100398 ret = register_virtio_device(vdev);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200399 if (ret) {
weiping zhang900a1632017-12-21 20:40:58 +0800400 put_device(&vdev->dev);
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200401 dev_err(dev, "failed to register vdev: %d\n", ret);
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100402 goto out;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200403 }
404
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100405 dev_info(dev, "registered %s (type %d)\n", dev_name(&vdev->dev), id);
406
407out:
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200408 return ret;
409}
410
411/**
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100412 * rproc_remove_virtio_dev() - remove an rproc-induced virtio device
Loic Pallardyd4c036f2019-01-21 14:55:15 +0100413 * @dev: the virtio device
414 * @data: must be null
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200415 *
Ohad Ben-Cohen7a186942012-02-13 22:30:39 +0100416 * This function unregisters an existing virtio device.
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200417 */
Loic Pallardyd4c036f2019-01-21 14:55:15 +0100418int rproc_remove_virtio_dev(struct device *dev, void *data)
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200419{
Loic Pallardyd4c036f2019-01-21 14:55:15 +0100420 struct virtio_device *vdev = dev_to_virtio(dev);
421
422 unregister_virtio_device(vdev);
423 return 0;
Ohad Ben-Cohenac8954a2011-10-20 18:15:13 +0200424}