incubator-nuttx/drivers/virtio/virtio-rpmb.c

290 lines
8.0 KiB
C

/****************************************************************************
* drivers/virtio/virtio-rpmb.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <debug.h>
#include <errno.h>
#include <stdio.h>
#include <nuttx/fs/fs.h>
#include <nuttx/fs/ioctl.h>
#include <nuttx/semaphore.h>
#include <nuttx/spinlock.h>
#include <nuttx/mmcsd.h>
#include <nuttx/virtio/virtio.h>
#include "virtio-rpmb.h"
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
#define VIRTIO_RPMB_DEVNAME "/dev/rpmb"
/****************************************************************************
* Private Types
****************************************************************************/
struct virtio_rpmb_priv_s
{
/* The virtio device we're associated with */
FAR struct virtio_device *vdev;
spinlock_t lock;
};
struct virtio_rpmb_cookie_s
{
sem_t sem;
size_t len;
};
/****************************************************************************
* Private Function Prototypes
****************************************************************************/
static int virtio_rpmb_probe(FAR struct virtio_device *vdev);
static void virtio_rpmb_remove(FAR struct virtio_device *vdev);
static int virtio_rpmb_ioctl(FAR struct file *, int, unsigned long);
/****************************************************************************
* Private Data
****************************************************************************/
static struct virtio_driver g_virtio_rpmb_driver =
{
LIST_INITIAL_VALUE(g_virtio_rpmb_driver.node), /* node */
VIRTIO_ID_RPMB, /* device id */
virtio_rpmb_probe, /* probe */
virtio_rpmb_remove, /* remove */
};
static const struct file_operations g_virtio_rpmb_ops =
{
NULL, /* open */
NULL, /* close */
NULL, /* read */
NULL, /* write */
NULL, /* seek */
virtio_rpmb_ioctl, /* ioctl */
NULL, /* mmap */
NULL, /* truncate */
NULL, /* poll */
#ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS
NULL, /* unlink */
#endif
};
/****************************************************************************
* Private Functions
****************************************************************************/
/* virtio_rpmb_done() - virtual queue completion callback */
static void virtio_rpmb_done(FAR struct virtqueue *vq)
{
FAR struct virtio_rpmb_priv_s *priv = vq->vq_dev->priv;
FAR struct virtio_rpmb_cookie_s *cookie;
uint32_t len;
for (; ; )
{
cookie = virtqueue_get_buffer_lock(vq, &len, NULL, &priv->lock);
if (cookie == NULL)
{
break;
}
/* Assign the return length */
cookie->len = len;
/* Read completed, post the sem */
nxsem_post(&cookie->sem);
}
}
/* This is just a simple helper for processing the virtqueue buf. It will
* block until the resp arrives. Returns number of bytes written back
* or negative if it failed.
*/
static int virtio_rpmb_transact(FAR struct virtio_rpmb_priv_s *priv,
FAR struct virtqueue_buf *vb,
int out, int in)
{
FAR struct virtqueue *vq = priv->vdev->vrings_info[0].vq;
struct virtio_rpmb_cookie_s cookie;
irqstate_t flags;
int ret;
/* Init the cookie */
cookie.len = 0;
nxsem_init(&cookie.sem, 0, 0);
/* Add the input buffer to the virtqueue, and the cookie as the virtqueue
* cookie. (virtqueue_get_buffer() will return cookie).
*/
flags = spin_lock_irqsave(&priv->lock);
ret = virtqueue_add_buffer(vq, vb, out, in, &cookie);
if (ret < 0)
{
spin_unlock_irqrestore(&priv->lock, flags);
return ret;
}
/* Notify the other side to process the added virtqueue buffer */
virtqueue_kick(vq);
spin_unlock_irqrestore(&priv->lock, flags);
/* Wait fot completion */
nxsem_wait_uninterruptible(&cookie.sem);
return cookie.len;
}
/****************************************************************************
* Name: virtio_blk_ioctl
****************************************************************************/
static int virtio_rpmb_ioctl(FAR struct file *filep, int cmd,
unsigned long arg)
{
FAR struct virtio_rpmb_priv_s *priv;
FAR struct mmc_ioc_multi_cmd *mioc = (FAR struct mmc_ioc_multi_cmd *)arg;
struct virtqueue_buf vb[mioc->num_of_cmds];
int ret;
int i;
priv = (FAR struct virtio_rpmb_priv_s *)filep->f_inode->i_private;
for (i = 0; i < mioc->num_of_cmds; i++)
{
vb[i].buf = (FAR void *)mioc->cmds[i].data_ptr;
vb[i].len = mioc->cmds[i].blocks * mioc->cmds[i].blksz;
}
ret = virtio_rpmb_transact(priv, vb, mioc->num_of_cmds - 1, 1);
return ret < 0 ? ret : 0;
}
static int virtio_rpmb_init(FAR struct virtio_rpmb_priv_s *priv,
FAR struct virtio_device *vdev)
{
FAR const char *vqname[1];
vq_callback callback[1];
int ret;
priv->vdev = vdev;
vdev->priv = priv;
spin_lock_init(&priv->lock);
/* Initialize the virtio device */
virtio_set_status(vdev, VIRTIO_CONFIG_STATUS_DRIVER);
virtio_set_features(vdev, 0);
virtio_set_status(vdev, VIRTIO_CONFIG_FEATURES_OK);
vqname[0] = "virtio_rpmb_vq";
callback[0] = virtio_rpmb_done;
ret = virtio_create_virtqueues(vdev, 0, 1, vqname, callback);
if (ret < 0)
{
vrterr("virtio_device_create_virtqueue failed, ret=%d\n", ret);
goto err_with_lock;
}
virtio_set_status(vdev, VIRTIO_CONFIG_STATUS_DRIVER_OK);
virtqueue_enable_cb(vdev->vrings_info[0].vq);
return OK;
err_with_lock:
return ret;
}
static void virtio_rpmb_uninit(FAR struct virtio_rpmb_priv_s *priv)
{
FAR struct virtio_device *vdev = priv->vdev;
virtio_reset_device(vdev);
virtio_delete_virtqueues(vdev);
}
static int virtio_rpmb_probe(FAR struct virtio_device *vdev)
{
FAR struct virtio_rpmb_priv_s *priv;
int ret;
/* Alloc the virtio block driver private data */
priv = kmm_zalloc(sizeof(*priv));
if (priv == NULL)
{
vrterr("No enough memory\n");
return -ENOMEM;
}
/* Init the virtio block driver */
ret = virtio_rpmb_init(priv, vdev);
if (ret < 0)
{
vrterr("virtio_rpmb_init failed, ret=%d\n", ret);
goto err_with_priv;
}
ret = register_driver(VIRTIO_RPMB_DEVNAME, &g_virtio_rpmb_ops, 0666,
priv);
if (ret < 0)
{
vrterr("Register NuttX driver failed, ret=%d\n", ret);
goto err_with_init;
}
return ret;
err_with_init:
virtio_rpmb_uninit(priv);
err_with_priv:
kmm_free(priv);
return ret;
}
static void virtio_rpmb_remove(FAR struct virtio_device *vdev)
{
FAR struct virtio_rpmb_priv_s *priv = vdev->priv;
unregister_driver(VIRTIO_RPMB_DEVNAME);
virtio_rpmb_uninit(priv);
kmm_free(priv);
}
int virtio_register_rpmb_driver(void)
{
return virtio_register_driver(&g_virtio_rpmb_driver);
}