#include <linux/crypto.h>
#include <linux/spinlock.h>
#include <linux/interrupt.h>
+#include <linux/workqueue.h>
#include <crypto/aead.h>
#include <crypto/aes.h>
#include <crypto/engine.h>
char name[32];
struct crypto_engine *engine;
- struct tasklet_struct done_task;
+ struct work_struct done_work;
};
struct virtio_crypto {
return 0;
}
-static void virtcrypto_done_task(unsigned long data)
+static void virtcrypto_done_work(struct work_struct *work)
{
- struct data_queue *data_vq = (struct data_queue *)data;
+ struct data_queue *data_vq = from_work(data_vq, work, done_work);
struct virtqueue *vq = data_vq->vq;
struct virtio_crypto_request *vc_req;
unsigned long flags;
struct virtio_crypto *vcrypto = vq->vdev->priv;
struct data_queue *dq = &vcrypto->data_vq[vq->index];
- tasklet_schedule(&dq->done_task);
+ queue_work(system_bh_wq, &dq->done_work);
}
static int virtcrypto_find_vqs(struct virtio_crypto *vi)
ret = -ENOMEM;
goto err_engine;
}
- tasklet_init(&vi->data_vq[i].done_task, virtcrypto_done_task,
- (unsigned long)&vi->data_vq[i]);
+ INIT_WORK(&vi->data_vq[i].done_work, virtcrypto_done_work);
}
kfree(vqs_info);
if (virtcrypto_dev_started(vcrypto))
virtcrypto_dev_stop(vcrypto);
for (i = 0; i < vcrypto->max_data_queues; i++)
- tasklet_kill(&vcrypto->data_vq[i].done_task);
+ cancel_work_sync(&vcrypto->data_vq[i].done_work);
virtio_reset_device(vdev);
virtcrypto_free_unused_reqs(vcrypto);
virtcrypto_clear_crypto_engines(vcrypto);