mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-07 13:53:24 +00:00
crypto: chelsio - Handle PCI shutdown event
chcr receives "CXGB4_STATE_DETACH" event on PCI Shutdown. Wait for processing of inflight request and Mark the device unavailable. Signed-off-by: Harsh Jain <harsh@chelsio.com> Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
This commit is contained in:
parent
c4f6d44d77
commit
fef4912b66
@ -123,7 +123,7 @@ static inline struct chcr_authenc_ctx *AUTHENC_CTX(struct chcr_aead_ctx *gctx)
|
|||||||
|
|
||||||
static inline struct uld_ctx *ULD_CTX(struct chcr_context *ctx)
|
static inline struct uld_ctx *ULD_CTX(struct chcr_context *ctx)
|
||||||
{
|
{
|
||||||
return ctx->dev->u_ctx;
|
return container_of(ctx->dev, struct uld_ctx, dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int is_ofld_imm(const struct sk_buff *skb)
|
static inline int is_ofld_imm(const struct sk_buff *skb)
|
||||||
@ -198,17 +198,40 @@ void chcr_verify_tag(struct aead_request *req, u8 *input, int *err)
|
|||||||
*err = 0;
|
*err = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int chcr_inc_wrcount(struct chcr_dev *dev)
|
||||||
|
{
|
||||||
|
int err = 0;
|
||||||
|
|
||||||
|
spin_lock_bh(&dev->lock_chcr_dev);
|
||||||
|
if (dev->state == CHCR_DETACH)
|
||||||
|
err = 1;
|
||||||
|
else
|
||||||
|
atomic_inc(&dev->inflight);
|
||||||
|
|
||||||
|
spin_unlock_bh(&dev->lock_chcr_dev);
|
||||||
|
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void chcr_dec_wrcount(struct chcr_dev *dev)
|
||||||
|
{
|
||||||
|
atomic_dec(&dev->inflight);
|
||||||
|
}
|
||||||
|
|
||||||
static inline void chcr_handle_aead_resp(struct aead_request *req,
|
static inline void chcr_handle_aead_resp(struct aead_request *req,
|
||||||
unsigned char *input,
|
unsigned char *input,
|
||||||
int err)
|
int err)
|
||||||
{
|
{
|
||||||
struct chcr_aead_reqctx *reqctx = aead_request_ctx(req);
|
struct chcr_aead_reqctx *reqctx = aead_request_ctx(req);
|
||||||
|
struct crypto_aead *tfm = crypto_aead_reqtfm(req);
|
||||||
|
struct chcr_dev *dev = a_ctx(tfm)->dev;
|
||||||
|
|
||||||
chcr_aead_common_exit(req);
|
chcr_aead_common_exit(req);
|
||||||
if (reqctx->verify == VERIFY_SW) {
|
if (reqctx->verify == VERIFY_SW) {
|
||||||
chcr_verify_tag(req, input, &err);
|
chcr_verify_tag(req, input, &err);
|
||||||
reqctx->verify = VERIFY_HW;
|
reqctx->verify = VERIFY_HW;
|
||||||
}
|
}
|
||||||
|
chcr_dec_wrcount(dev);
|
||||||
req->base.complete(&req->base, err);
|
req->base.complete(&req->base, err);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1100,6 +1123,7 @@ static int chcr_handle_cipher_resp(struct ablkcipher_request *req,
|
|||||||
struct cpl_fw6_pld *fw6_pld = (struct cpl_fw6_pld *)input;
|
struct cpl_fw6_pld *fw6_pld = (struct cpl_fw6_pld *)input;
|
||||||
struct chcr_blkcipher_req_ctx *reqctx = ablkcipher_request_ctx(req);
|
struct chcr_blkcipher_req_ctx *reqctx = ablkcipher_request_ctx(req);
|
||||||
struct cipher_wr_param wrparam;
|
struct cipher_wr_param wrparam;
|
||||||
|
struct chcr_dev *dev = c_ctx(tfm)->dev;
|
||||||
int bytes;
|
int bytes;
|
||||||
|
|
||||||
if (err)
|
if (err)
|
||||||
@ -1161,6 +1185,7 @@ static int chcr_handle_cipher_resp(struct ablkcipher_request *req,
|
|||||||
unmap:
|
unmap:
|
||||||
chcr_cipher_dma_unmap(&ULD_CTX(c_ctx(tfm))->lldi.pdev->dev, req);
|
chcr_cipher_dma_unmap(&ULD_CTX(c_ctx(tfm))->lldi.pdev->dev, req);
|
||||||
complete:
|
complete:
|
||||||
|
chcr_dec_wrcount(dev);
|
||||||
req->base.complete(&req->base, err);
|
req->base.complete(&req->base, err);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
@ -1187,7 +1212,10 @@ static int process_cipher(struct ablkcipher_request *req,
|
|||||||
ablkctx->enckey_len, req->nbytes, ivsize);
|
ablkctx->enckey_len, req->nbytes, ivsize);
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
chcr_cipher_dma_map(&ULD_CTX(c_ctx(tfm))->lldi.pdev->dev, req);
|
|
||||||
|
err = chcr_cipher_dma_map(&ULD_CTX(c_ctx(tfm))->lldi.pdev->dev, req);
|
||||||
|
if (err)
|
||||||
|
goto error;
|
||||||
if (req->nbytes < (SGE_MAX_WR_LEN - (sizeof(struct chcr_wr) +
|
if (req->nbytes < (SGE_MAX_WR_LEN - (sizeof(struct chcr_wr) +
|
||||||
AES_MIN_KEY_SIZE +
|
AES_MIN_KEY_SIZE +
|
||||||
sizeof(struct cpl_rx_phys_dsgl) +
|
sizeof(struct cpl_rx_phys_dsgl) +
|
||||||
@ -1276,15 +1304,21 @@ static int process_cipher(struct ablkcipher_request *req,
|
|||||||
static int chcr_aes_encrypt(struct ablkcipher_request *req)
|
static int chcr_aes_encrypt(struct ablkcipher_request *req)
|
||||||
{
|
{
|
||||||
struct crypto_ablkcipher *tfm = crypto_ablkcipher_reqtfm(req);
|
struct crypto_ablkcipher *tfm = crypto_ablkcipher_reqtfm(req);
|
||||||
|
struct chcr_dev *dev = c_ctx(tfm)->dev;
|
||||||
struct sk_buff *skb = NULL;
|
struct sk_buff *skb = NULL;
|
||||||
int err, isfull = 0;
|
int err, isfull = 0;
|
||||||
struct uld_ctx *u_ctx = ULD_CTX(c_ctx(tfm));
|
struct uld_ctx *u_ctx = ULD_CTX(c_ctx(tfm));
|
||||||
|
|
||||||
|
err = chcr_inc_wrcount(dev);
|
||||||
|
if (err)
|
||||||
|
return -ENXIO;
|
||||||
if (unlikely(cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
if (unlikely(cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
||||||
c_ctx(tfm)->tx_qidx))) {
|
c_ctx(tfm)->tx_qidx))) {
|
||||||
isfull = 1;
|
isfull = 1;
|
||||||
if (!(req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG))
|
if (!(req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG)) {
|
||||||
return -ENOSPC;
|
err = -ENOSPC;
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
err = process_cipher(req, u_ctx->lldi.rxq_ids[c_ctx(tfm)->rx_qidx],
|
err = process_cipher(req, u_ctx->lldi.rxq_ids[c_ctx(tfm)->rx_qidx],
|
||||||
@ -1295,15 +1329,23 @@ static int chcr_aes_encrypt(struct ablkcipher_request *req)
|
|||||||
set_wr_txq(skb, CPL_PRIORITY_DATA, c_ctx(tfm)->tx_qidx);
|
set_wr_txq(skb, CPL_PRIORITY_DATA, c_ctx(tfm)->tx_qidx);
|
||||||
chcr_send_wr(skb);
|
chcr_send_wr(skb);
|
||||||
return isfull ? -EBUSY : -EINPROGRESS;
|
return isfull ? -EBUSY : -EINPROGRESS;
|
||||||
|
error:
|
||||||
|
chcr_dec_wrcount(dev);
|
||||||
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int chcr_aes_decrypt(struct ablkcipher_request *req)
|
static int chcr_aes_decrypt(struct ablkcipher_request *req)
|
||||||
{
|
{
|
||||||
struct crypto_ablkcipher *tfm = crypto_ablkcipher_reqtfm(req);
|
struct crypto_ablkcipher *tfm = crypto_ablkcipher_reqtfm(req);
|
||||||
struct uld_ctx *u_ctx = ULD_CTX(c_ctx(tfm));
|
struct uld_ctx *u_ctx = ULD_CTX(c_ctx(tfm));
|
||||||
|
struct chcr_dev *dev = c_ctx(tfm)->dev;
|
||||||
struct sk_buff *skb = NULL;
|
struct sk_buff *skb = NULL;
|
||||||
int err, isfull = 0;
|
int err, isfull = 0;
|
||||||
|
|
||||||
|
err = chcr_inc_wrcount(dev);
|
||||||
|
if (err)
|
||||||
|
return -ENXIO;
|
||||||
|
|
||||||
if (unlikely(cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
if (unlikely(cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
||||||
c_ctx(tfm)->tx_qidx))) {
|
c_ctx(tfm)->tx_qidx))) {
|
||||||
isfull = 1;
|
isfull = 1;
|
||||||
@ -1333,10 +1375,11 @@ static int chcr_device_init(struct chcr_context *ctx)
|
|||||||
if (!ctx->dev) {
|
if (!ctx->dev) {
|
||||||
u_ctx = assign_chcr_device();
|
u_ctx = assign_chcr_device();
|
||||||
if (!u_ctx) {
|
if (!u_ctx) {
|
||||||
|
err = -ENXIO;
|
||||||
pr_err("chcr device assignment fails\n");
|
pr_err("chcr device assignment fails\n");
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
ctx->dev = u_ctx->dev;
|
ctx->dev = &u_ctx->dev;
|
||||||
adap = padap(ctx->dev);
|
adap = padap(ctx->dev);
|
||||||
ntxq = u_ctx->lldi.ntxq;
|
ntxq = u_ctx->lldi.ntxq;
|
||||||
rxq_perchan = u_ctx->lldi.nrxq / u_ctx->lldi.nchan;
|
rxq_perchan = u_ctx->lldi.nrxq / u_ctx->lldi.nchan;
|
||||||
@ -1561,6 +1604,7 @@ static int chcr_ahash_update(struct ahash_request *req)
|
|||||||
struct chcr_ahash_req_ctx *req_ctx = ahash_request_ctx(req);
|
struct chcr_ahash_req_ctx *req_ctx = ahash_request_ctx(req);
|
||||||
struct crypto_ahash *rtfm = crypto_ahash_reqtfm(req);
|
struct crypto_ahash *rtfm = crypto_ahash_reqtfm(req);
|
||||||
struct uld_ctx *u_ctx = NULL;
|
struct uld_ctx *u_ctx = NULL;
|
||||||
|
struct chcr_dev *dev = h_ctx(rtfm)->dev;
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
u8 remainder = 0, bs;
|
u8 remainder = 0, bs;
|
||||||
unsigned int nbytes = req->nbytes;
|
unsigned int nbytes = req->nbytes;
|
||||||
@ -1569,12 +1613,6 @@ static int chcr_ahash_update(struct ahash_request *req)
|
|||||||
|
|
||||||
bs = crypto_tfm_alg_blocksize(crypto_ahash_tfm(rtfm));
|
bs = crypto_tfm_alg_blocksize(crypto_ahash_tfm(rtfm));
|
||||||
u_ctx = ULD_CTX(h_ctx(rtfm));
|
u_ctx = ULD_CTX(h_ctx(rtfm));
|
||||||
if (unlikely(cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
|
||||||
h_ctx(rtfm)->tx_qidx))) {
|
|
||||||
isfull = 1;
|
|
||||||
if (!(req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG))
|
|
||||||
return -ENOSPC;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (nbytes + req_ctx->reqlen >= bs) {
|
if (nbytes + req_ctx->reqlen >= bs) {
|
||||||
remainder = (nbytes + req_ctx->reqlen) % bs;
|
remainder = (nbytes + req_ctx->reqlen) % bs;
|
||||||
@ -1585,10 +1623,27 @@ static int chcr_ahash_update(struct ahash_request *req)
|
|||||||
req_ctx->reqlen += nbytes;
|
req_ctx->reqlen += nbytes;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
error = chcr_inc_wrcount(dev);
|
||||||
|
if (error)
|
||||||
|
return -ENXIO;
|
||||||
|
/* Detach state for CHCR means lldi or padap is freed. Increasing
|
||||||
|
* inflight count for dev guarantees that lldi and padap is valid
|
||||||
|
*/
|
||||||
|
if (unlikely(cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
||||||
|
h_ctx(rtfm)->tx_qidx))) {
|
||||||
|
isfull = 1;
|
||||||
|
if (!(req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG)) {
|
||||||
|
error = -ENOSPC;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
chcr_init_hctx_per_wr(req_ctx);
|
chcr_init_hctx_per_wr(req_ctx);
|
||||||
error = chcr_hash_dma_map(&u_ctx->lldi.pdev->dev, req);
|
error = chcr_hash_dma_map(&u_ctx->lldi.pdev->dev, req);
|
||||||
if (error)
|
if (error) {
|
||||||
return -ENOMEM;
|
error = -ENOMEM;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
get_alg_config(¶ms.alg_prm, crypto_ahash_digestsize(rtfm));
|
get_alg_config(¶ms.alg_prm, crypto_ahash_digestsize(rtfm));
|
||||||
params.kctx_len = roundup(params.alg_prm.result_size, 16);
|
params.kctx_len = roundup(params.alg_prm.result_size, 16);
|
||||||
params.sg_len = chcr_hash_ent_in_wr(req->src, !!req_ctx->reqlen,
|
params.sg_len = chcr_hash_ent_in_wr(req->src, !!req_ctx->reqlen,
|
||||||
@ -1628,6 +1683,8 @@ static int chcr_ahash_update(struct ahash_request *req)
|
|||||||
return isfull ? -EBUSY : -EINPROGRESS;
|
return isfull ? -EBUSY : -EINPROGRESS;
|
||||||
unmap:
|
unmap:
|
||||||
chcr_hash_dma_unmap(&u_ctx->lldi.pdev->dev, req);
|
chcr_hash_dma_unmap(&u_ctx->lldi.pdev->dev, req);
|
||||||
|
err:
|
||||||
|
chcr_dec_wrcount(dev);
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1645,10 +1702,16 @@ static int chcr_ahash_final(struct ahash_request *req)
|
|||||||
{
|
{
|
||||||
struct chcr_ahash_req_ctx *req_ctx = ahash_request_ctx(req);
|
struct chcr_ahash_req_ctx *req_ctx = ahash_request_ctx(req);
|
||||||
struct crypto_ahash *rtfm = crypto_ahash_reqtfm(req);
|
struct crypto_ahash *rtfm = crypto_ahash_reqtfm(req);
|
||||||
|
struct chcr_dev *dev = h_ctx(rtfm)->dev;
|
||||||
struct hash_wr_param params;
|
struct hash_wr_param params;
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
struct uld_ctx *u_ctx = NULL;
|
struct uld_ctx *u_ctx = NULL;
|
||||||
u8 bs = crypto_tfm_alg_blocksize(crypto_ahash_tfm(rtfm));
|
u8 bs = crypto_tfm_alg_blocksize(crypto_ahash_tfm(rtfm));
|
||||||
|
int error = -EINVAL;
|
||||||
|
|
||||||
|
error = chcr_inc_wrcount(dev);
|
||||||
|
if (error)
|
||||||
|
return -ENXIO;
|
||||||
|
|
||||||
chcr_init_hctx_per_wr(req_ctx);
|
chcr_init_hctx_per_wr(req_ctx);
|
||||||
u_ctx = ULD_CTX(h_ctx(rtfm));
|
u_ctx = ULD_CTX(h_ctx(rtfm));
|
||||||
@ -1685,19 +1748,25 @@ static int chcr_ahash_final(struct ahash_request *req)
|
|||||||
}
|
}
|
||||||
params.hash_size = crypto_ahash_digestsize(rtfm);
|
params.hash_size = crypto_ahash_digestsize(rtfm);
|
||||||
skb = create_hash_wr(req, ¶ms);
|
skb = create_hash_wr(req, ¶ms);
|
||||||
if (IS_ERR(skb))
|
if (IS_ERR(skb)) {
|
||||||
return PTR_ERR(skb);
|
error = PTR_ERR(skb);
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
req_ctx->reqlen = 0;
|
req_ctx->reqlen = 0;
|
||||||
skb->dev = u_ctx->lldi.ports[0];
|
skb->dev = u_ctx->lldi.ports[0];
|
||||||
set_wr_txq(skb, CPL_PRIORITY_DATA, h_ctx(rtfm)->tx_qidx);
|
set_wr_txq(skb, CPL_PRIORITY_DATA, h_ctx(rtfm)->tx_qidx);
|
||||||
chcr_send_wr(skb);
|
chcr_send_wr(skb);
|
||||||
return -EINPROGRESS;
|
return -EINPROGRESS;
|
||||||
|
err:
|
||||||
|
chcr_dec_wrcount(dev);
|
||||||
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int chcr_ahash_finup(struct ahash_request *req)
|
static int chcr_ahash_finup(struct ahash_request *req)
|
||||||
{
|
{
|
||||||
struct chcr_ahash_req_ctx *req_ctx = ahash_request_ctx(req);
|
struct chcr_ahash_req_ctx *req_ctx = ahash_request_ctx(req);
|
||||||
struct crypto_ahash *rtfm = crypto_ahash_reqtfm(req);
|
struct crypto_ahash *rtfm = crypto_ahash_reqtfm(req);
|
||||||
|
struct chcr_dev *dev = h_ctx(rtfm)->dev;
|
||||||
struct uld_ctx *u_ctx = NULL;
|
struct uld_ctx *u_ctx = NULL;
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
struct hash_wr_param params;
|
struct hash_wr_param params;
|
||||||
@ -1706,17 +1775,24 @@ static int chcr_ahash_finup(struct ahash_request *req)
|
|||||||
|
|
||||||
bs = crypto_tfm_alg_blocksize(crypto_ahash_tfm(rtfm));
|
bs = crypto_tfm_alg_blocksize(crypto_ahash_tfm(rtfm));
|
||||||
u_ctx = ULD_CTX(h_ctx(rtfm));
|
u_ctx = ULD_CTX(h_ctx(rtfm));
|
||||||
|
error = chcr_inc_wrcount(dev);
|
||||||
|
if (error)
|
||||||
|
return -ENXIO;
|
||||||
|
|
||||||
if (unlikely(cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
if (unlikely(cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
||||||
h_ctx(rtfm)->tx_qidx))) {
|
h_ctx(rtfm)->tx_qidx))) {
|
||||||
isfull = 1;
|
isfull = 1;
|
||||||
if (!(req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG))
|
if (!(req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG)) {
|
||||||
return -ENOSPC;
|
error = -ENOSPC;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
chcr_init_hctx_per_wr(req_ctx);
|
chcr_init_hctx_per_wr(req_ctx);
|
||||||
error = chcr_hash_dma_map(&u_ctx->lldi.pdev->dev, req);
|
error = chcr_hash_dma_map(&u_ctx->lldi.pdev->dev, req);
|
||||||
if (error)
|
if (error) {
|
||||||
return -ENOMEM;
|
error = -ENOMEM;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
get_alg_config(¶ms.alg_prm, crypto_ahash_digestsize(rtfm));
|
get_alg_config(¶ms.alg_prm, crypto_ahash_digestsize(rtfm));
|
||||||
params.kctx_len = roundup(params.alg_prm.result_size, 16);
|
params.kctx_len = roundup(params.alg_prm.result_size, 16);
|
||||||
@ -1773,6 +1849,8 @@ static int chcr_ahash_finup(struct ahash_request *req)
|
|||||||
return isfull ? -EBUSY : -EINPROGRESS;
|
return isfull ? -EBUSY : -EINPROGRESS;
|
||||||
unmap:
|
unmap:
|
||||||
chcr_hash_dma_unmap(&u_ctx->lldi.pdev->dev, req);
|
chcr_hash_dma_unmap(&u_ctx->lldi.pdev->dev, req);
|
||||||
|
err:
|
||||||
|
chcr_dec_wrcount(dev);
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1780,6 +1858,7 @@ static int chcr_ahash_digest(struct ahash_request *req)
|
|||||||
{
|
{
|
||||||
struct chcr_ahash_req_ctx *req_ctx = ahash_request_ctx(req);
|
struct chcr_ahash_req_ctx *req_ctx = ahash_request_ctx(req);
|
||||||
struct crypto_ahash *rtfm = crypto_ahash_reqtfm(req);
|
struct crypto_ahash *rtfm = crypto_ahash_reqtfm(req);
|
||||||
|
struct chcr_dev *dev = h_ctx(rtfm)->dev;
|
||||||
struct uld_ctx *u_ctx = NULL;
|
struct uld_ctx *u_ctx = NULL;
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
struct hash_wr_param params;
|
struct hash_wr_param params;
|
||||||
@ -1788,19 +1867,26 @@ static int chcr_ahash_digest(struct ahash_request *req)
|
|||||||
|
|
||||||
rtfm->init(req);
|
rtfm->init(req);
|
||||||
bs = crypto_tfm_alg_blocksize(crypto_ahash_tfm(rtfm));
|
bs = crypto_tfm_alg_blocksize(crypto_ahash_tfm(rtfm));
|
||||||
|
error = chcr_inc_wrcount(dev);
|
||||||
|
if (error)
|
||||||
|
return -ENXIO;
|
||||||
|
|
||||||
u_ctx = ULD_CTX(h_ctx(rtfm));
|
u_ctx = ULD_CTX(h_ctx(rtfm));
|
||||||
if (unlikely(cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
if (unlikely(cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
||||||
h_ctx(rtfm)->tx_qidx))) {
|
h_ctx(rtfm)->tx_qidx))) {
|
||||||
isfull = 1;
|
isfull = 1;
|
||||||
if (!(req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG))
|
if (!(req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG)) {
|
||||||
return -ENOSPC;
|
error = -ENOSPC;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
chcr_init_hctx_per_wr(req_ctx);
|
chcr_init_hctx_per_wr(req_ctx);
|
||||||
error = chcr_hash_dma_map(&u_ctx->lldi.pdev->dev, req);
|
error = chcr_hash_dma_map(&u_ctx->lldi.pdev->dev, req);
|
||||||
if (error)
|
if (error) {
|
||||||
return -ENOMEM;
|
error = -ENOMEM;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
get_alg_config(¶ms.alg_prm, crypto_ahash_digestsize(rtfm));
|
get_alg_config(¶ms.alg_prm, crypto_ahash_digestsize(rtfm));
|
||||||
params.kctx_len = roundup(params.alg_prm.result_size, 16);
|
params.kctx_len = roundup(params.alg_prm.result_size, 16);
|
||||||
@ -1853,6 +1939,8 @@ static int chcr_ahash_digest(struct ahash_request *req)
|
|||||||
return isfull ? -EBUSY : -EINPROGRESS;
|
return isfull ? -EBUSY : -EINPROGRESS;
|
||||||
unmap:
|
unmap:
|
||||||
chcr_hash_dma_unmap(&u_ctx->lldi.pdev->dev, req);
|
chcr_hash_dma_unmap(&u_ctx->lldi.pdev->dev, req);
|
||||||
|
err:
|
||||||
|
chcr_dec_wrcount(dev);
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1924,6 +2012,7 @@ static inline void chcr_handle_ahash_resp(struct ahash_request *req,
|
|||||||
int digestsize, updated_digestsize;
|
int digestsize, updated_digestsize;
|
||||||
struct crypto_ahash *tfm = crypto_ahash_reqtfm(req);
|
struct crypto_ahash *tfm = crypto_ahash_reqtfm(req);
|
||||||
struct uld_ctx *u_ctx = ULD_CTX(h_ctx(tfm));
|
struct uld_ctx *u_ctx = ULD_CTX(h_ctx(tfm));
|
||||||
|
struct chcr_dev *dev = h_ctx(tfm)->dev;
|
||||||
|
|
||||||
if (input == NULL)
|
if (input == NULL)
|
||||||
goto out;
|
goto out;
|
||||||
@ -1966,6 +2055,7 @@ static inline void chcr_handle_ahash_resp(struct ahash_request *req,
|
|||||||
|
|
||||||
|
|
||||||
out:
|
out:
|
||||||
|
chcr_dec_wrcount(dev);
|
||||||
req->base.complete(&req->base, err);
|
req->base.complete(&req->base, err);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3553,27 +3643,42 @@ static int chcr_aead_op(struct aead_request *req,
|
|||||||
create_wr_t create_wr_fn)
|
create_wr_t create_wr_fn)
|
||||||
{
|
{
|
||||||
struct crypto_aead *tfm = crypto_aead_reqtfm(req);
|
struct crypto_aead *tfm = crypto_aead_reqtfm(req);
|
||||||
|
struct chcr_aead_reqctx *reqctx = aead_request_ctx(req);
|
||||||
struct uld_ctx *u_ctx;
|
struct uld_ctx *u_ctx;
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
int isfull = 0;
|
int isfull = 0;
|
||||||
|
struct chcr_dev *cdev;
|
||||||
|
|
||||||
if (!a_ctx(tfm)->dev) {
|
cdev = a_ctx(tfm)->dev;
|
||||||
|
if (!cdev) {
|
||||||
pr_err("chcr : %s : No crypto device.\n", __func__);
|
pr_err("chcr : %s : No crypto device.\n", __func__);
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (chcr_inc_wrcount(cdev)) {
|
||||||
|
/* Detach state for CHCR means lldi or padap is freed.
|
||||||
|
* We cannot increment fallback here.
|
||||||
|
*/
|
||||||
|
return chcr_aead_fallback(req, reqctx->op);
|
||||||
|
}
|
||||||
|
|
||||||
u_ctx = ULD_CTX(a_ctx(tfm));
|
u_ctx = ULD_CTX(a_ctx(tfm));
|
||||||
if (cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
if (cxgb4_is_crypto_q_full(u_ctx->lldi.ports[0],
|
||||||
a_ctx(tfm)->tx_qidx)) {
|
a_ctx(tfm)->tx_qidx)) {
|
||||||
isfull = 1;
|
isfull = 1;
|
||||||
if (!(req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG))
|
if (!(req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG)) {
|
||||||
|
chcr_dec_wrcount(cdev);
|
||||||
return -ENOSPC;
|
return -ENOSPC;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Form a WR from req */
|
/* Form a WR from req */
|
||||||
skb = create_wr_fn(req, u_ctx->lldi.rxq_ids[a_ctx(tfm)->rx_qidx], size);
|
skb = create_wr_fn(req, u_ctx->lldi.rxq_ids[a_ctx(tfm)->rx_qidx], size);
|
||||||
|
|
||||||
if (IS_ERR(skb) || !skb)
|
if (IS_ERR(skb) || !skb) {
|
||||||
|
chcr_dec_wrcount(cdev);
|
||||||
return PTR_ERR(skb);
|
return PTR_ERR(skb);
|
||||||
|
}
|
||||||
|
|
||||||
skb->dev = u_ctx->lldi.ports[0];
|
skb->dev = u_ctx->lldi.ports[0];
|
||||||
set_wr_txq(skb, CPL_PRIORITY_DATA, a_ctx(tfm)->tx_qidx);
|
set_wr_txq(skb, CPL_PRIORITY_DATA, a_ctx(tfm)->tx_qidx);
|
||||||
|
@ -26,10 +26,7 @@
|
|||||||
#include "chcr_core.h"
|
#include "chcr_core.h"
|
||||||
#include "cxgb4_uld.h"
|
#include "cxgb4_uld.h"
|
||||||
|
|
||||||
static LIST_HEAD(uld_ctx_list);
|
static struct chcr_driver_data drv_data;
|
||||||
static DEFINE_MUTEX(dev_mutex);
|
|
||||||
static atomic_t dev_count;
|
|
||||||
static struct uld_ctx *ctx_rr;
|
|
||||||
|
|
||||||
typedef int (*chcr_handler_func)(struct chcr_dev *dev, unsigned char *input);
|
typedef int (*chcr_handler_func)(struct chcr_dev *dev, unsigned char *input);
|
||||||
static int cpl_fw6_pld_handler(struct chcr_dev *dev, unsigned char *input);
|
static int cpl_fw6_pld_handler(struct chcr_dev *dev, unsigned char *input);
|
||||||
@ -53,6 +50,29 @@ static struct cxgb4_uld_info chcr_uld_info = {
|
|||||||
#endif /* CONFIG_CHELSIO_IPSEC_INLINE */
|
#endif /* CONFIG_CHELSIO_IPSEC_INLINE */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void detach_work_fn(struct work_struct *work)
|
||||||
|
{
|
||||||
|
struct chcr_dev *dev;
|
||||||
|
|
||||||
|
dev = container_of(work, struct chcr_dev, detach_work.work);
|
||||||
|
|
||||||
|
if (atomic_read(&dev->inflight)) {
|
||||||
|
dev->wqretry--;
|
||||||
|
if (dev->wqretry) {
|
||||||
|
pr_debug("Request Inflight Count %d\n",
|
||||||
|
atomic_read(&dev->inflight));
|
||||||
|
|
||||||
|
schedule_delayed_work(&dev->detach_work, WQ_DETACH_TM);
|
||||||
|
} else {
|
||||||
|
WARN(1, "CHCR:%d request Still Pending\n",
|
||||||
|
atomic_read(&dev->inflight));
|
||||||
|
complete(&dev->detach_comp);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
complete(&dev->detach_comp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
struct uld_ctx *assign_chcr_device(void)
|
struct uld_ctx *assign_chcr_device(void)
|
||||||
{
|
{
|
||||||
struct uld_ctx *u_ctx = NULL;
|
struct uld_ctx *u_ctx = NULL;
|
||||||
@ -63,56 +83,70 @@ struct uld_ctx *assign_chcr_device(void)
|
|||||||
* Although One session must use the same device to
|
* Although One session must use the same device to
|
||||||
* maintain request-response ordering.
|
* maintain request-response ordering.
|
||||||
*/
|
*/
|
||||||
mutex_lock(&dev_mutex);
|
mutex_lock(&drv_data.drv_mutex);
|
||||||
if (!list_empty(&uld_ctx_list)) {
|
if (!list_empty(&drv_data.act_dev)) {
|
||||||
u_ctx = ctx_rr;
|
u_ctx = drv_data.last_dev;
|
||||||
if (list_is_last(&ctx_rr->entry, &uld_ctx_list))
|
if (list_is_last(&drv_data.last_dev->entry, &drv_data.act_dev))
|
||||||
ctx_rr = list_first_entry(&uld_ctx_list,
|
drv_data.last_dev = list_first_entry(&drv_data.act_dev,
|
||||||
struct uld_ctx,
|
struct uld_ctx, entry);
|
||||||
entry);
|
|
||||||
else
|
else
|
||||||
ctx_rr = list_next_entry(ctx_rr, entry);
|
drv_data.last_dev =
|
||||||
|
list_next_entry(drv_data.last_dev, entry);
|
||||||
}
|
}
|
||||||
mutex_unlock(&dev_mutex);
|
mutex_unlock(&drv_data.drv_mutex);
|
||||||
return u_ctx;
|
return u_ctx;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int chcr_dev_add(struct uld_ctx *u_ctx)
|
static void chcr_dev_add(struct uld_ctx *u_ctx)
|
||||||
{
|
{
|
||||||
struct chcr_dev *dev;
|
struct chcr_dev *dev;
|
||||||
|
|
||||||
dev = kzalloc(sizeof(*dev), GFP_KERNEL);
|
dev = &u_ctx->dev;
|
||||||
if (!dev)
|
dev->state = CHCR_ATTACH;
|
||||||
return -ENXIO;
|
atomic_set(&dev->inflight, 0);
|
||||||
|
mutex_lock(&drv_data.drv_mutex);
|
||||||
spin_lock_init(&dev->lock_chcr_dev);
|
list_move(&u_ctx->entry, &drv_data.act_dev);
|
||||||
u_ctx->dev = dev;
|
if (!drv_data.last_dev)
|
||||||
dev->u_ctx = u_ctx;
|
drv_data.last_dev = u_ctx;
|
||||||
atomic_inc(&dev_count);
|
mutex_unlock(&drv_data.drv_mutex);
|
||||||
mutex_lock(&dev_mutex);
|
|
||||||
list_add_tail(&u_ctx->entry, &uld_ctx_list);
|
|
||||||
if (!ctx_rr)
|
|
||||||
ctx_rr = u_ctx;
|
|
||||||
mutex_unlock(&dev_mutex);
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int chcr_dev_remove(struct uld_ctx *u_ctx)
|
static void chcr_dev_init(struct uld_ctx *u_ctx)
|
||||||
{
|
{
|
||||||
if (ctx_rr == u_ctx) {
|
struct chcr_dev *dev;
|
||||||
if (list_is_last(&ctx_rr->entry, &uld_ctx_list))
|
|
||||||
ctx_rr = list_first_entry(&uld_ctx_list,
|
dev = &u_ctx->dev;
|
||||||
struct uld_ctx,
|
spin_lock_init(&dev->lock_chcr_dev);
|
||||||
entry);
|
INIT_DELAYED_WORK(&dev->detach_work, detach_work_fn);
|
||||||
|
init_completion(&dev->detach_comp);
|
||||||
|
dev->state = CHCR_INIT;
|
||||||
|
dev->wqretry = WQ_RETRY;
|
||||||
|
atomic_inc(&drv_data.dev_count);
|
||||||
|
atomic_set(&dev->inflight, 0);
|
||||||
|
mutex_lock(&drv_data.drv_mutex);
|
||||||
|
list_add_tail(&u_ctx->entry, &drv_data.inact_dev);
|
||||||
|
if (!drv_data.last_dev)
|
||||||
|
drv_data.last_dev = u_ctx;
|
||||||
|
mutex_unlock(&drv_data.drv_mutex);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int chcr_dev_move(struct uld_ctx *u_ctx)
|
||||||
|
{
|
||||||
|
mutex_lock(&drv_data.drv_mutex);
|
||||||
|
if (drv_data.last_dev == u_ctx) {
|
||||||
|
if (list_is_last(&drv_data.last_dev->entry, &drv_data.act_dev))
|
||||||
|
drv_data.last_dev = list_first_entry(&drv_data.act_dev,
|
||||||
|
struct uld_ctx, entry);
|
||||||
else
|
else
|
||||||
ctx_rr = list_next_entry(ctx_rr, entry);
|
drv_data.last_dev =
|
||||||
|
list_next_entry(drv_data.last_dev, entry);
|
||||||
}
|
}
|
||||||
list_del(&u_ctx->entry);
|
list_move(&u_ctx->entry, &drv_data.inact_dev);
|
||||||
if (list_empty(&uld_ctx_list))
|
if (list_empty(&drv_data.act_dev))
|
||||||
ctx_rr = NULL;
|
drv_data.last_dev = NULL;
|
||||||
kfree(u_ctx->dev);
|
atomic_dec(&drv_data.dev_count);
|
||||||
u_ctx->dev = NULL;
|
mutex_unlock(&drv_data.drv_mutex);
|
||||||
atomic_dec(&dev_count);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -167,6 +201,7 @@ static void *chcr_uld_add(const struct cxgb4_lld_info *lld)
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
u_ctx->lldi = *lld;
|
u_ctx->lldi = *lld;
|
||||||
|
chcr_dev_init(u_ctx);
|
||||||
#ifdef CONFIG_CHELSIO_IPSEC_INLINE
|
#ifdef CONFIG_CHELSIO_IPSEC_INLINE
|
||||||
if (lld->crypto & ULP_CRYPTO_IPSEC_INLINE)
|
if (lld->crypto & ULP_CRYPTO_IPSEC_INLINE)
|
||||||
chcr_add_xfrmops(lld);
|
chcr_add_xfrmops(lld);
|
||||||
@ -179,7 +214,7 @@ int chcr_uld_rx_handler(void *handle, const __be64 *rsp,
|
|||||||
const struct pkt_gl *pgl)
|
const struct pkt_gl *pgl)
|
||||||
{
|
{
|
||||||
struct uld_ctx *u_ctx = (struct uld_ctx *)handle;
|
struct uld_ctx *u_ctx = (struct uld_ctx *)handle;
|
||||||
struct chcr_dev *dev = u_ctx->dev;
|
struct chcr_dev *dev = &u_ctx->dev;
|
||||||
const struct cpl_fw6_pld *rpl = (struct cpl_fw6_pld *)rsp;
|
const struct cpl_fw6_pld *rpl = (struct cpl_fw6_pld *)rsp;
|
||||||
|
|
||||||
if (rpl->opcode != CPL_FW6_PLD) {
|
if (rpl->opcode != CPL_FW6_PLD) {
|
||||||
@ -201,6 +236,28 @@ int chcr_uld_tx_handler(struct sk_buff *skb, struct net_device *dev)
|
|||||||
}
|
}
|
||||||
#endif /* CONFIG_CHELSIO_IPSEC_INLINE */
|
#endif /* CONFIG_CHELSIO_IPSEC_INLINE */
|
||||||
|
|
||||||
|
static void chcr_detach_device(struct uld_ctx *u_ctx)
|
||||||
|
{
|
||||||
|
struct chcr_dev *dev = &u_ctx->dev;
|
||||||
|
|
||||||
|
spin_lock_bh(&dev->lock_chcr_dev);
|
||||||
|
if (dev->state == CHCR_DETACH) {
|
||||||
|
spin_unlock_bh(&dev->lock_chcr_dev);
|
||||||
|
pr_debug("Detached Event received for already detach device\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
dev->state = CHCR_DETACH;
|
||||||
|
spin_unlock_bh(&dev->lock_chcr_dev);
|
||||||
|
|
||||||
|
if (atomic_read(&dev->inflight) != 0) {
|
||||||
|
schedule_delayed_work(&dev->detach_work, WQ_DETACH_TM);
|
||||||
|
wait_for_completion(&dev->detach_comp);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Move u_ctx to inactive_dev list
|
||||||
|
chcr_dev_move(u_ctx);
|
||||||
|
}
|
||||||
|
|
||||||
static int chcr_uld_state_change(void *handle, enum cxgb4_state state)
|
static int chcr_uld_state_change(void *handle, enum cxgb4_state state)
|
||||||
{
|
{
|
||||||
struct uld_ctx *u_ctx = handle;
|
struct uld_ctx *u_ctx = handle;
|
||||||
@ -208,23 +265,16 @@ static int chcr_uld_state_change(void *handle, enum cxgb4_state state)
|
|||||||
|
|
||||||
switch (state) {
|
switch (state) {
|
||||||
case CXGB4_STATE_UP:
|
case CXGB4_STATE_UP:
|
||||||
if (!u_ctx->dev) {
|
if (u_ctx->dev.state != CHCR_INIT) {
|
||||||
ret = chcr_dev_add(u_ctx);
|
// ALready Initialised.
|
||||||
if (ret != 0)
|
return 0;
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
if (atomic_read(&dev_count) == 1)
|
chcr_dev_add(u_ctx);
|
||||||
ret = start_crypto();
|
ret = start_crypto();
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case CXGB4_STATE_DETACH:
|
case CXGB4_STATE_DETACH:
|
||||||
if (u_ctx->dev) {
|
chcr_detach_device(u_ctx);
|
||||||
mutex_lock(&dev_mutex);
|
|
||||||
chcr_dev_remove(u_ctx);
|
|
||||||
mutex_unlock(&dev_mutex);
|
|
||||||
}
|
|
||||||
if (!atomic_read(&dev_count))
|
|
||||||
stop_crypto();
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case CXGB4_STATE_START_RECOVERY:
|
case CXGB4_STATE_START_RECOVERY:
|
||||||
@ -237,7 +287,13 @@ static int chcr_uld_state_change(void *handle, enum cxgb4_state state)
|
|||||||
|
|
||||||
static int __init chcr_crypto_init(void)
|
static int __init chcr_crypto_init(void)
|
||||||
{
|
{
|
||||||
|
INIT_LIST_HEAD(&drv_data.act_dev);
|
||||||
|
INIT_LIST_HEAD(&drv_data.inact_dev);
|
||||||
|
atomic_set(&drv_data.dev_count, 0);
|
||||||
|
mutex_init(&drv_data.drv_mutex);
|
||||||
|
drv_data.last_dev = NULL;
|
||||||
cxgb4_register_uld(CXGB4_ULD_CRYPTO, &chcr_uld_info);
|
cxgb4_register_uld(CXGB4_ULD_CRYPTO, &chcr_uld_info);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -245,18 +301,20 @@ static void __exit chcr_crypto_exit(void)
|
|||||||
{
|
{
|
||||||
struct uld_ctx *u_ctx, *tmp;
|
struct uld_ctx *u_ctx, *tmp;
|
||||||
|
|
||||||
if (atomic_read(&dev_count))
|
stop_crypto();
|
||||||
stop_crypto();
|
|
||||||
|
|
||||||
|
cxgb4_unregister_uld(CXGB4_ULD_CRYPTO);
|
||||||
/* Remove all devices from list */
|
/* Remove all devices from list */
|
||||||
mutex_lock(&dev_mutex);
|
mutex_lock(&drv_data.drv_mutex);
|
||||||
list_for_each_entry_safe(u_ctx, tmp, &uld_ctx_list, entry) {
|
list_for_each_entry_safe(u_ctx, tmp, &drv_data.act_dev, entry) {
|
||||||
if (u_ctx->dev)
|
list_del(&u_ctx->entry);
|
||||||
chcr_dev_remove(u_ctx);
|
|
||||||
kfree(u_ctx);
|
kfree(u_ctx);
|
||||||
}
|
}
|
||||||
mutex_unlock(&dev_mutex);
|
list_for_each_entry_safe(u_ctx, tmp, &drv_data.inact_dev, entry) {
|
||||||
cxgb4_unregister_uld(CXGB4_ULD_CRYPTO);
|
list_del(&u_ctx->entry);
|
||||||
|
kfree(u_ctx);
|
||||||
|
}
|
||||||
|
mutex_unlock(&drv_data.drv_mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
module_init(chcr_crypto_init);
|
module_init(chcr_crypto_init);
|
||||||
|
@ -47,7 +47,7 @@
|
|||||||
|
|
||||||
#define MAX_PENDING_REQ_TO_HW 20
|
#define MAX_PENDING_REQ_TO_HW 20
|
||||||
#define CHCR_TEST_RESPONSE_TIMEOUT 1000
|
#define CHCR_TEST_RESPONSE_TIMEOUT 1000
|
||||||
|
#define WQ_DETACH_TM (msecs_to_jiffies(50))
|
||||||
#define PAD_ERROR_BIT 1
|
#define PAD_ERROR_BIT 1
|
||||||
#define CHK_PAD_ERR_BIT(x) (((x) >> PAD_ERROR_BIT) & 1)
|
#define CHK_PAD_ERR_BIT(x) (((x) >> PAD_ERROR_BIT) & 1)
|
||||||
|
|
||||||
@ -61,9 +61,6 @@
|
|||||||
#define HASH_WR_MIN_LEN (sizeof(struct chcr_wr) + \
|
#define HASH_WR_MIN_LEN (sizeof(struct chcr_wr) + \
|
||||||
DUMMY_BYTES + \
|
DUMMY_BYTES + \
|
||||||
sizeof(struct ulptx_sgl))
|
sizeof(struct ulptx_sgl))
|
||||||
|
|
||||||
#define padap(dev) pci_get_drvdata(dev->u_ctx->lldi.pdev)
|
|
||||||
|
|
||||||
struct uld_ctx;
|
struct uld_ctx;
|
||||||
|
|
||||||
struct _key_ctx {
|
struct _key_ctx {
|
||||||
@ -121,6 +118,20 @@ struct _key_ctx {
|
|||||||
#define KEYCTX_TX_WR_AUTHIN_G(x) \
|
#define KEYCTX_TX_WR_AUTHIN_G(x) \
|
||||||
(((x) >> KEYCTX_TX_WR_AUTHIN_S) & KEYCTX_TX_WR_AUTHIN_M)
|
(((x) >> KEYCTX_TX_WR_AUTHIN_S) & KEYCTX_TX_WR_AUTHIN_M)
|
||||||
|
|
||||||
|
#define WQ_RETRY 5
|
||||||
|
struct chcr_driver_data {
|
||||||
|
struct list_head act_dev;
|
||||||
|
struct list_head inact_dev;
|
||||||
|
atomic_t dev_count;
|
||||||
|
struct mutex drv_mutex;
|
||||||
|
struct uld_ctx *last_dev;
|
||||||
|
};
|
||||||
|
|
||||||
|
enum chcr_state {
|
||||||
|
CHCR_INIT = 0,
|
||||||
|
CHCR_ATTACH,
|
||||||
|
CHCR_DETACH,
|
||||||
|
};
|
||||||
struct chcr_wr {
|
struct chcr_wr {
|
||||||
struct fw_crypto_lookaside_wr wreq;
|
struct fw_crypto_lookaside_wr wreq;
|
||||||
struct ulp_txpkt ulptx;
|
struct ulp_txpkt ulptx;
|
||||||
@ -131,14 +142,18 @@ struct chcr_wr {
|
|||||||
|
|
||||||
struct chcr_dev {
|
struct chcr_dev {
|
||||||
spinlock_t lock_chcr_dev;
|
spinlock_t lock_chcr_dev;
|
||||||
struct uld_ctx *u_ctx;
|
enum chcr_state state;
|
||||||
|
atomic_t inflight;
|
||||||
|
int wqretry;
|
||||||
|
struct delayed_work detach_work;
|
||||||
|
struct completion detach_comp;
|
||||||
unsigned char tx_channel_id;
|
unsigned char tx_channel_id;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct uld_ctx {
|
struct uld_ctx {
|
||||||
struct list_head entry;
|
struct list_head entry;
|
||||||
struct cxgb4_lld_info lldi;
|
struct cxgb4_lld_info lldi;
|
||||||
struct chcr_dev *dev;
|
struct chcr_dev dev;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct sge_opaque_hdr {
|
struct sge_opaque_hdr {
|
||||||
@ -189,6 +204,13 @@ static inline unsigned int sgl_len(unsigned int n)
|
|||||||
return (3 * n) / 2 + (n & 1) + 2;
|
return (3 * n) / 2 + (n & 1) + 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void *padap(struct chcr_dev *dev)
|
||||||
|
{
|
||||||
|
struct uld_ctx *u_ctx = container_of(dev, struct uld_ctx, dev);
|
||||||
|
|
||||||
|
return pci_get_drvdata(u_ctx->lldi.pdev);
|
||||||
|
}
|
||||||
|
|
||||||
struct uld_ctx *assign_chcr_device(void);
|
struct uld_ctx *assign_chcr_device(void);
|
||||||
int chcr_send_wr(struct sk_buff *skb);
|
int chcr_send_wr(struct sk_buff *skb);
|
||||||
int start_crypto(void);
|
int start_crypto(void);
|
||||||
|
Loading…
Reference in New Issue
Block a user