#include <crypto/algapi.h>
 #include <crypto/hash.h>
 #include <crypto/md5.h>
+#include <crypto/sm3.h>
 #include <crypto/internal/hash.h>
 
 #include "cc_driver.h"
 
 #define CC_MAX_HASH_SEQ_LEN 12
 #define CC_MAX_OPAD_KEYS_SIZE CC_MAX_HASH_BLCK_SIZE
+#define CC_SM3_HASH_LEN_SIZE 8
 
 struct cc_hash_handle {
        cc_sram_addr_t digest_len_sram_addr; /* const value in SRAM*/
 static u64 sha512_init[] = {
        SHA512_H7, SHA512_H6, SHA512_H5, SHA512_H4,
        SHA512_H3, SHA512_H2, SHA512_H1, SHA512_H0 };
+static const u32 sm3_init[] = {
+       SM3_IVH, SM3_IVG, SM3_IVF, SM3_IVE,
+       SM3_IVD, SM3_IVC, SM3_IVB, SM3_IVA };
 
 static void cc_setup_xcbc(struct ahash_request *areq, struct cc_hw_desc desc[],
                          unsigned int *seq_size);
 {
        struct cc_hash_ctx *ctx = crypto_tfm_ctx(tfm);
 
-       return cc_get_default_hash_len(ctx->drvdata);
+       if (ctx->hash_mode == DRV_HASH_SM3)
+               return CC_SM3_HASH_LEN_SIZE;
+       else
+               return cc_get_default_hash_len(ctx->drvdata);
 }
 
 static int cc_cra_init(struct crypto_tfm *tfm)
        char mac_name[CRYPTO_MAX_ALG_NAME];
        char mac_driver_name[CRYPTO_MAX_ALG_NAME];
        unsigned int blocksize;
+       bool is_mac;
        bool synchronize;
        struct ahash_alg template_ahash;
        int hash_mode;
                .mac_name = "hmac(sha1)",
                .mac_driver_name = "hmac-sha1-ccree",
                .blocksize = SHA1_BLOCK_SIZE,
+               .is_mac = true,
                .synchronize = false,
                .template_ahash = {
                        .init = cc_hash_init,
                .mac_name = "hmac(sha256)",
                .mac_driver_name = "hmac-sha256-ccree",
                .blocksize = SHA256_BLOCK_SIZE,
+               .is_mac = true,
                .template_ahash = {
                        .init = cc_hash_init,
                        .update = cc_hash_update,
                .mac_name = "hmac(sha224)",
                .mac_driver_name = "hmac-sha224-ccree",
                .blocksize = SHA224_BLOCK_SIZE,
+               .is_mac = true,
                .template_ahash = {
                        .init = cc_hash_init,
                        .update = cc_hash_update,
                .mac_name = "hmac(sha384)",
                .mac_driver_name = "hmac-sha384-ccree",
                .blocksize = SHA384_BLOCK_SIZE,
+               .is_mac = true,
                .template_ahash = {
                        .init = cc_hash_init,
                        .update = cc_hash_update,
                .mac_name = "hmac(sha512)",
                .mac_driver_name = "hmac-sha512-ccree",
                .blocksize = SHA512_BLOCK_SIZE,
+               .is_mac = true,
                .template_ahash = {
                        .init = cc_hash_init,
                        .update = cc_hash_update,
                .mac_name = "hmac(md5)",
                .mac_driver_name = "hmac-md5-ccree",
                .blocksize = MD5_HMAC_BLOCK_SIZE,
+               .is_mac = true,
                .template_ahash = {
                        .init = cc_hash_init,
                        .update = cc_hash_update,
                .inter_digestsize = MD5_DIGEST_SIZE,
                .min_hw_rev = CC_HW_REV_630,
        },
+       {
+               .name = "sm3",
+               .driver_name = "sm3-ccree",
+               .blocksize = SM3_BLOCK_SIZE,
+               .is_mac = false,
+               .template_ahash = {
+                       .init = cc_hash_init,
+                       .update = cc_hash_update,
+                       .final = cc_hash_final,
+                       .finup = cc_hash_finup,
+                       .digest = cc_hash_digest,
+                       .export = cc_hash_export,
+                       .import = cc_hash_import,
+                       .setkey = cc_hash_setkey,
+                       .halg = {
+                               .digestsize = SM3_DIGEST_SIZE,
+                               .statesize = CC_STATE_SIZE(SM3_DIGEST_SIZE),
+                       },
+               },
+               .hash_mode = DRV_HASH_SM3,
+               .hw_mode = DRV_HASH_HW_SM3,
+               .inter_digestsize = SM3_DIGEST_SIZE,
+               .min_hw_rev = CC_HW_REV_713,
+       },
        {
                .mac_name = "xcbc(aes)",
                .mac_driver_name = "xcbc-aes-ccree",
                .blocksize = AES_BLOCK_SIZE,
+               .is_mac = true,
                .template_ahash = {
                        .init = cc_hash_init,
                        .update = cc_mac_update,
                .mac_name = "cmac(aes)",
                .mac_driver_name = "cmac-aes-ccree",
                .blocksize = AES_BLOCK_SIZE,
+               .is_mac = true,
                .template_ahash = {
                        .init = cc_hash_init,
                        .update = cc_mac_update,
        unsigned int larval_seq_len = 0;
        struct cc_hw_desc larval_seq[CC_DIGEST_SIZE_MAX / sizeof(u32)];
        bool large_sha_supported = (drvdata->hw_rev >= CC_HW_REV_712);
+       bool sm3_supported = (drvdata->hw_rev >= CC_HW_REV_713);
        int rc = 0;
 
        /* Copy-to-sram digest-len */
        sram_buff_ofs += sizeof(sha256_init);
        larval_seq_len = 0;
 
+       if (sm3_supported) {
+               cc_set_sram_desc(sm3_init, sram_buff_ofs,
+                                ARRAY_SIZE(sm3_init), larval_seq,
+                                &larval_seq_len);
+               rc = send_request_init(drvdata, larval_seq, larval_seq_len);
+               if (rc)
+                       goto init_digest_const_err;
+               sram_buff_ofs += sizeof(sm3_init);
+               larval_seq_len = 0;
+       }
+
        if (large_sha_supported) {
                cc_set_sram_desc((u32 *)sha384_init, sram_buff_ofs,
                                 (ARRAY_SIZE(sha384_init) * 2), larval_seq,
                        sizeof(sha224_init) +
                        sizeof(sha256_init);
 
+       if (drvdata->hw_rev >= CC_HW_REV_713)
+               sram_size_to_alloc += sizeof(sm3_init);
+
        if (drvdata->hw_rev >= CC_HW_REV_712)
                sram_size_to_alloc += sizeof(digest_len_sha512_init) +
                        sizeof(sha384_init) + sizeof(sha512_init);
                /* We either support both HASH and MAC or none */
                if (driver_hash[alg].min_hw_rev > drvdata->hw_rev)
                        continue;
-
-               /* register hmac version */
-               t_alg = cc_alloc_hash_alg(&driver_hash[alg], dev, true);
-               if (IS_ERR(t_alg)) {
-                       rc = PTR_ERR(t_alg);
-                       dev_err(dev, "%s alg allocation failed\n",
-                               driver_hash[alg].driver_name);
-                       goto fail;
-               }
-               t_alg->drvdata = drvdata;
-
-               rc = crypto_register_ahash(&t_alg->ahash_alg);
-               if (rc) {
-                       dev_err(dev, "%s alg registration failed\n",
-                               driver_hash[alg].driver_name);
-                       kfree(t_alg);
-                       goto fail;
-               } else {
-                       list_add_tail(&t_alg->entry, &hash_handle->hash_list);
+               if (driver_hash[alg].is_mac) {
+                       /* register hmac version */
+                       t_alg = cc_alloc_hash_alg(&driver_hash[alg], dev, true);
+                       if (IS_ERR(t_alg)) {
+                               rc = PTR_ERR(t_alg);
+                               dev_err(dev, "%s alg allocation failed\n",
+                                       driver_hash[alg].driver_name);
+                               goto fail;
+                       }
+                       t_alg->drvdata = drvdata;
+
+                       rc = crypto_register_ahash(&t_alg->ahash_alg);
+                       if (rc) {
+                               dev_err(dev, "%s alg registration failed\n",
+                                       driver_hash[alg].driver_name);
+                               kfree(t_alg);
+                               goto fail;
+                       } else {
+                               list_add_tail(&t_alg->entry,
+                                             &hash_handle->hash_list);
+                       }
                }
-
                if (hw_mode == DRV_CIPHER_XCBC_MAC ||
                    hw_mode == DRV_CIPHER_CMAC)
                        continue;
                return sha384_init;
        case DRV_HASH_SHA512:
                return sha512_init;
+       case DRV_HASH_SM3:
+               return sm3_init;
        default:
                dev_err(dev, "Invalid hash mode (%d)\n", mode);
                return md5_init;
        struct cc_drvdata *_drvdata = (struct cc_drvdata *)drvdata;
        struct cc_hash_handle *hash_handle = _drvdata->hash_handle;
        struct device *dev = drvdata_to_dev(_drvdata);
+       bool sm3_supported = (_drvdata->hw_rev >= CC_HW_REV_713);
+       cc_sram_addr_t addr;
 
        switch (mode) {
        case DRV_HASH_NULL:
                        sizeof(md5_init) +
                        sizeof(sha1_init) +
                        sizeof(sha224_init));
-       case DRV_HASH_SHA384:
+       case DRV_HASH_SM3:
                return (hash_handle->larval_digest_sram_addr +
                        sizeof(md5_init) +
                        sizeof(sha1_init) +
                        sizeof(sha224_init) +
                        sizeof(sha256_init));
+       case DRV_HASH_SHA384:
+               addr = (hash_handle->larval_digest_sram_addr +
+                       sizeof(md5_init) +
+                       sizeof(sha1_init) +
+                       sizeof(sha224_init) +
+                       sizeof(sha256_init));
+               if (sm3_supported)
+                       addr += sizeof(sm3_init);
+               return addr;
        case DRV_HASH_SHA512:
-               return (hash_handle->larval_digest_sram_addr +
+               addr = (hash_handle->larval_digest_sram_addr +
                        sizeof(md5_init) +
                        sizeof(sha1_init) +
                        sizeof(sha224_init) +
                        sizeof(sha256_init) +
                        sizeof(sha384_init));
+               if (sm3_supported)
+                       addr += sizeof(sm3_init);
+               return addr;
        default:
                dev_err(dev, "Invalid hash mode (%d)\n", mode);
        }