2016-12-19 10:20:44 +08:00
/*
* Driver for EIP97 cryptographic accelerator .
*
* Copyright ( c ) 2016 Ryder Lee < ryder . lee @ mediatek . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
*/
# ifndef __MTK_PLATFORM_H_
# define __MTK_PLATFORM_H_
# include <crypto/algapi.h>
2017-01-20 13:41:15 +08:00
# include <crypto/internal/aead.h>
2016-12-19 10:20:44 +08:00
# include <crypto/internal/hash.h>
# include <crypto/scatterwalk.h>
2017-01-20 13:41:15 +08:00
# include <crypto/skcipher.h>
2016-12-19 10:20:44 +08:00
# include <linux/crypto.h>
# include <linux/dma-mapping.h>
# include <linux/interrupt.h>
# include <linux/scatterlist.h>
# include "mtk-regs.h"
# define MTK_RDR_PROC_THRESH BIT(0)
# define MTK_RDR_PROC_MODE BIT(23)
# define MTK_CNT_RST BIT(31)
# define MTK_IRQ_RDR0 BIT(1)
# define MTK_IRQ_RDR1 BIT(3)
# define MTK_IRQ_RDR2 BIT(5)
# define MTK_IRQ_RDR3 BIT(7)
# define SIZE_IN_WORDS(x) ((x) >> 2)
/**
* Ring 0 / 1 are used by AES encrypt and decrypt .
* Ring 2 / 3 are used by SHA .
*/
enum {
2017-03-09 10:11:13 +08:00
MTK_RING0 ,
MTK_RING1 ,
MTK_RING2 ,
MTK_RING3 ,
MTK_RING_MAX
2016-12-19 10:20:44 +08:00
} ;
2017-03-09 10:11:13 +08:00
# define MTK_REC_NUM (MTK_RING_MAX / 2)
2016-12-19 10:20:44 +08:00
# define MTK_IRQ_NUM 5
/**
* struct mtk_desc - DMA descriptor
* @ hdr : the descriptor control header
* @ buf : DMA address of input buffer segment
* @ ct : DMA address of command token that control operation flow
* @ ct_hdr : the command token control header
* @ tag : the user - defined field
* @ tfm : DMA address of transform state
* @ bound : align descriptors offset boundary
*
* Structure passed to the crypto engine to describe where source
* data needs to be fetched and how it needs to be processed .
*/
struct mtk_desc {
__le32 hdr ;
__le32 buf ;
__le32 ct ;
__le32 ct_hdr ;
__le32 tag ;
__le32 tfm ;
__le32 bound [ 2 ] ;
} ;
# define MTK_DESC_NUM 512
# define MTK_DESC_OFF SIZE_IN_WORDS(sizeof(struct mtk_desc))
# define MTK_DESC_SZ (MTK_DESC_OFF - 2)
# define MTK_DESC_RING_SZ ((sizeof(struct mtk_desc) * MTK_DESC_NUM))
# define MTK_DESC_CNT(x) ((MTK_DESC_OFF * (x)) << 2)
# define MTK_DESC_LAST cpu_to_le32(BIT(22))
# define MTK_DESC_FIRST cpu_to_le32(BIT(23))
# define MTK_DESC_BUF_LEN(x) cpu_to_le32(x)
# define MTK_DESC_CT_LEN(x) cpu_to_le32((x) << 24)
/**
* struct mtk_ring - Descriptor ring
* @ cmd_base : pointer to command descriptor ring base
2017-03-09 10:11:15 +08:00
* @ cmd_next : pointer to the next command descriptor
2016-12-19 10:20:44 +08:00
* @ cmd_dma : DMA address of command descriptor ring
* @ res_base : pointer to result descriptor ring base
2017-03-09 10:11:15 +08:00
* @ res_next : pointer to the next result descriptor
2017-03-09 10:11:18 +08:00
* @ res_prev : pointer to the previous result descriptor
2016-12-19 10:20:44 +08:00
* @ res_dma : DMA address of result descriptor ring
*
* A descriptor ring is a circular buffer that is used to manage
* one or more descriptors . There are two type of descriptor rings ;
* the command descriptor ring and result descriptor ring .
*/
struct mtk_ring {
struct mtk_desc * cmd_base ;
2017-03-09 10:11:15 +08:00
struct mtk_desc * cmd_next ;
2016-12-19 10:20:44 +08:00
dma_addr_t cmd_dma ;
struct mtk_desc * res_base ;
2017-03-09 10:11:15 +08:00
struct mtk_desc * res_next ;
2017-03-09 10:11:18 +08:00
struct mtk_desc * res_prev ;
2016-12-19 10:20:44 +08:00
dma_addr_t res_dma ;
} ;
/**
* struct mtk_aes_dma - Structure that holds sg list info
* @ sg : pointer to scatter - gather list
* @ nents : number of entries in the sg list
* @ remainder : remainder of sg list
* @ sg_len : number of entries in the sg mapped list
*/
struct mtk_aes_dma {
struct scatterlist * sg ;
int nents ;
u32 remainder ;
u32 sg_len ;
} ;
2017-01-20 13:41:10 +08:00
struct mtk_aes_base_ctx ;
struct mtk_aes_rec ;
struct mtk_cryp ;
typedef int ( * mtk_aes_fn ) ( struct mtk_cryp * cryp , struct mtk_aes_rec * aes ) ;
2017-01-20 13:41:08 +08:00
2016-12-19 10:20:44 +08:00
/**
* struct mtk_aes_rec - AES operation record
2017-03-09 10:11:12 +08:00
* @ cryp : pointer to Cryptographic device
2016-12-19 10:20:44 +08:00
* @ queue : crypto request queue
2017-01-20 13:41:13 +08:00
* @ areq : pointer to async request
2017-03-09 10:11:16 +08:00
* @ done_task : the tasklet is use in AES interrupt
* @ queue_task : the tasklet is used to dequeue request
2017-01-20 13:41:08 +08:00
* @ ctx : pointer to current context
2016-12-19 10:20:44 +08:00
* @ src : the structure that holds source sg list info
* @ dst : the structure that holds destination sg list info
* @ aligned_sg : the scatter list is use to alignment
* @ real_dst : pointer to the destination sg list
2017-01-20 13:41:11 +08:00
* @ resume : pointer to resume function
2016-12-19 10:20:44 +08:00
* @ total : request buffer length
* @ buf : pointer to page buffer
2017-03-09 10:11:13 +08:00
* @ id : the current use of ring
2016-12-19 10:20:44 +08:00
* @ flags : it ' s describing AES operation state
2017-01-20 13:41:10 +08:00
* @ lock : the async queue lock
2016-12-19 10:20:44 +08:00
*
* Structure used to record AES execution state .
*/
struct mtk_aes_rec {
2017-03-09 10:11:12 +08:00
struct mtk_cryp * cryp ;
2016-12-19 10:20:44 +08:00
struct crypto_queue queue ;
2017-01-20 13:41:10 +08:00
struct crypto_async_request * areq ;
2017-03-09 10:11:16 +08:00
struct tasklet_struct done_task ;
struct tasklet_struct queue_task ;
2017-01-20 13:41:10 +08:00
struct mtk_aes_base_ctx * ctx ;
2016-12-19 10:20:44 +08:00
struct mtk_aes_dma src ;
struct mtk_aes_dma dst ;
struct scatterlist aligned_sg ;
struct scatterlist * real_dst ;
2017-01-20 13:41:11 +08:00
mtk_aes_fn resume ;
2016-12-19 10:20:44 +08:00
size_t total ;
void * buf ;
u8 id ;
unsigned long flags ;
/* queue lock */
spinlock_t lock ;
} ;
/**
* struct mtk_sha_rec - SHA operation record
2017-03-09 10:11:12 +08:00
* @ cryp : pointer to Cryptographic device
2016-12-19 10:20:44 +08:00
* @ queue : crypto request queue
* @ req : pointer to ahash request
2017-03-09 10:11:16 +08:00
* @ done_task : the tasklet is use in SHA interrupt
* @ queue_task : the tasklet is used to dequeue request
2017-03-09 10:11:13 +08:00
* @ id : the current use of ring
2016-12-19 10:20:44 +08:00
* @ flags : it ' s describing SHA operation state
2017-03-09 10:11:13 +08:00
* @ lock : the async queue lock
2016-12-19 10:20:44 +08:00
*
* Structure used to record SHA execution state .
*/
struct mtk_sha_rec {
2017-03-09 10:11:12 +08:00
struct mtk_cryp * cryp ;
2016-12-19 10:20:44 +08:00
struct crypto_queue queue ;
struct ahash_request * req ;
2017-03-09 10:11:16 +08:00
struct tasklet_struct done_task ;
struct tasklet_struct queue_task ;
2016-12-19 10:20:44 +08:00
u8 id ;
unsigned long flags ;
/* queue lock */
spinlock_t lock ;
} ;
/**
* struct mtk_cryp - Cryptographic device
* @ base : pointer to mapped register I / O base
* @ dev : pointer to device
* @ clk_cryp : pointer to crypto clock
* @ irq : global system and rings IRQ
2017-03-09 10:11:13 +08:00
* @ ring : pointer to descriptor rings
* @ aes : pointer to operation record of AES
* @ sha : pointer to operation record of SHA
2016-12-19 10:20:44 +08:00
* @ aes_list : device list of AES
* @ sha_list : device list of SHA
* @ rec : it ' s used to select SHA record for tfm
*
* Structure storing cryptographic device information .
*/
struct mtk_cryp {
void __iomem * base ;
struct device * dev ;
struct clk * clk_cryp ;
int irq [ MTK_IRQ_NUM ] ;
2017-03-09 10:11:13 +08:00
struct mtk_ring * ring [ MTK_RING_MAX ] ;
2016-12-19 10:20:44 +08:00
struct mtk_aes_rec * aes [ MTK_REC_NUM ] ;
struct mtk_sha_rec * sha [ MTK_REC_NUM ] ;
struct list_head aes_list ;
struct list_head sha_list ;
bool rec ;
} ;
int mtk_cipher_alg_register ( struct mtk_cryp * cryp ) ;
void mtk_cipher_alg_release ( struct mtk_cryp * cryp ) ;
int mtk_hash_alg_register ( struct mtk_cryp * cryp ) ;
void mtk_hash_alg_release ( struct mtk_cryp * cryp ) ;
# endif /* __MTK_PLATFORM_H_ */