Commit 093d5615 authored by Gilad Ben-Yossef's avatar Gilad Ben-Yossef Committed by Greg Kroah-Hartman

staging: ccree: simplify resource release on error

The resource release on probe/init error was being handled
in an awkward manner and possibly leaking memory on certain
(unlikely) error path.

Fix it by simplifying the error resource release and making
it easier to track.
Reported-by: default avatarDan Carpenter <dan.carpenter@oracle.com>
Signed-off-by: default avatarGilad Ben-Yossef <gilad@benyossef.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 7f6f832d
...@@ -2720,6 +2720,7 @@ int ssi_aead_alloc(struct ssi_drvdata *drvdata) ...@@ -2720,6 +2720,7 @@ int ssi_aead_alloc(struct ssi_drvdata *drvdata)
goto fail0; goto fail0;
} }
INIT_LIST_HEAD(&aead_handle->aead_list);
drvdata->aead_handle = aead_handle; drvdata->aead_handle = aead_handle;
aead_handle->sram_workspace_addr = ssi_sram_mgr_alloc( aead_handle->sram_workspace_addr = ssi_sram_mgr_alloc(
...@@ -2730,8 +2731,6 @@ int ssi_aead_alloc(struct ssi_drvdata *drvdata) ...@@ -2730,8 +2731,6 @@ int ssi_aead_alloc(struct ssi_drvdata *drvdata)
goto fail1; goto fail1;
} }
INIT_LIST_HEAD(&aead_handle->aead_list);
/* Linux crypto */ /* Linux crypto */
for (alg = 0; alg < ARRAY_SIZE(aead_algs); alg++) { for (alg = 0; alg < ARRAY_SIZE(aead_algs); alg++) {
t_alg = ssi_aead_create_alg(&aead_algs[alg]); t_alg = ssi_aead_create_alg(&aead_algs[alg]);
......
...@@ -1312,9 +1312,8 @@ int ssi_ablkcipher_alloc(struct ssi_drvdata *drvdata) ...@@ -1312,9 +1312,8 @@ int ssi_ablkcipher_alloc(struct ssi_drvdata *drvdata)
if (!ablkcipher_handle) if (!ablkcipher_handle)
return -ENOMEM; return -ENOMEM;
drvdata->blkcipher_handle = ablkcipher_handle;
INIT_LIST_HEAD(&ablkcipher_handle->blkcipher_alg_list); INIT_LIST_HEAD(&ablkcipher_handle->blkcipher_alg_list);
drvdata->blkcipher_handle = ablkcipher_handle;
/* Linux crypto */ /* Linux crypto */
SSI_LOG_DEBUG("Number of algorithms = %zu\n", ARRAY_SIZE(blkcipher_algs)); SSI_LOG_DEBUG("Number of algorithms = %zu\n", ARRAY_SIZE(blkcipher_algs));
......
...@@ -233,16 +233,14 @@ static int init_cc_resources(struct platform_device *plat_dev) ...@@ -233,16 +233,14 @@ static int init_cc_resources(struct platform_device *plat_dev)
if (!new_drvdata) { if (!new_drvdata) {
SSI_LOG_ERR("Failed to allocate drvdata"); SSI_LOG_ERR("Failed to allocate drvdata");
rc = -ENOMEM; rc = -ENOMEM;
goto init_cc_res_err; goto post_drvdata_err;
} }
dev_set_drvdata(&plat_dev->dev, new_drvdata);
new_drvdata->plat_dev = plat_dev;
new_drvdata->clk = of_clk_get(np, 0); new_drvdata->clk = of_clk_get(np, 0);
new_drvdata->coherent = of_dma_is_coherent(np); new_drvdata->coherent = of_dma_is_coherent(np);
/*Initialize inflight counter used in dx_ablkcipher_secure_complete used for count of BYSPASS blocks operations*/
new_drvdata->inflight_counter = 0;
dev_set_drvdata(&plat_dev->dev, new_drvdata);
/* Get device resources */ /* Get device resources */
/* First CC registers space */ /* First CC registers space */
req_mem_cc_regs = platform_get_resource(plat_dev, IORESOURCE_MEM, 0); req_mem_cc_regs = platform_get_resource(plat_dev, IORESOURCE_MEM, 0);
...@@ -250,38 +248,42 @@ static int init_cc_resources(struct platform_device *plat_dev) ...@@ -250,38 +248,42 @@ static int init_cc_resources(struct platform_device *plat_dev)
new_drvdata->cc_base = devm_ioremap_resource(&plat_dev->dev, new_drvdata->cc_base = devm_ioremap_resource(&plat_dev->dev,
req_mem_cc_regs); req_mem_cc_regs);
if (IS_ERR(new_drvdata->cc_base)) { if (IS_ERR(new_drvdata->cc_base)) {
SSI_LOG_ERR("Failed to ioremap registers");
rc = PTR_ERR(new_drvdata->cc_base); rc = PTR_ERR(new_drvdata->cc_base);
goto init_cc_res_err; goto post_drvdata_err;
} }
SSI_LOG_DEBUG("Got MEM resource (%s): start=%pad end=%pad\n", SSI_LOG_DEBUG("Got MEM resource (%s): start=%pad end=%pad\n",
req_mem_cc_regs->name, req_mem_cc_regs->name,
req_mem_cc_regs->start, req_mem_cc_regs->start,
req_mem_cc_regs->end); req_mem_cc_regs->end);
SSI_LOG_DEBUG("CC registers mapped from %pa to 0x%p\n", SSI_LOG_DEBUG("CC registers mapped from %pa to 0x%p\n",
&req_mem_cc_regs->start, new_drvdata->cc_base); &req_mem_cc_regs->start, new_drvdata->cc_base);
cc_base = new_drvdata->cc_base; cc_base = new_drvdata->cc_base;
/* Then IRQ */ /* Then IRQ */
new_drvdata->irq = platform_get_irq(plat_dev, 0); new_drvdata->irq = platform_get_irq(plat_dev, 0);
if (new_drvdata->irq < 0) { if (new_drvdata->irq < 0) {
SSI_LOG_ERR("Failed getting IRQ resource\n"); SSI_LOG_ERR("Failed getting IRQ resource\n");
rc = new_drvdata->irq; rc = new_drvdata->irq;
goto init_cc_res_err; goto post_drvdata_err;
} }
rc = devm_request_irq(&plat_dev->dev, new_drvdata->irq, cc_isr, rc = devm_request_irq(&plat_dev->dev, new_drvdata->irq, cc_isr,
IRQF_SHARED, "arm_cc7x", new_drvdata); IRQF_SHARED, "arm_cc7x", new_drvdata);
if (rc) { if (rc) {
SSI_LOG_ERR("Could not register to interrupt %d\n", SSI_LOG_ERR("Could not register to interrupt %d\n",
new_drvdata->irq); new_drvdata->irq);
goto init_cc_res_err; goto post_drvdata_err;
} }
init_completion(&new_drvdata->icache_setup_completion);
SSI_LOG_DEBUG("Registered to IRQ: %d\n", new_drvdata->irq); SSI_LOG_DEBUG("Registered to IRQ: %d\n", new_drvdata->irq);
new_drvdata->plat_dev = plat_dev;
init_completion(&new_drvdata->icache_setup_completion);
rc = cc_clk_on(new_drvdata); rc = cc_clk_on(new_drvdata);
if (rc) if (rc)
goto init_cc_res_err; goto post_drvdata_err;
if (!new_drvdata->plat_dev->dev.dma_mask) if (!new_drvdata->plat_dev->dev.dma_mask)
new_drvdata->plat_dev->dev.dma_mask = &new_drvdata->plat_dev->dev.coherent_dma_mask; new_drvdata->plat_dev->dev.dma_mask = &new_drvdata->plat_dev->dev.coherent_dma_mask;
...@@ -295,7 +297,7 @@ static int init_cc_resources(struct platform_device *plat_dev) ...@@ -295,7 +297,7 @@ static int init_cc_resources(struct platform_device *plat_dev)
SSI_LOG_ERR("Invalid CC signature: SIGNATURE=0x%08X != expected=0x%08X\n", SSI_LOG_ERR("Invalid CC signature: SIGNATURE=0x%08X != expected=0x%08X\n",
signature_val, (u32)DX_DEV_SIGNATURE); signature_val, (u32)DX_DEV_SIGNATURE);
rc = -EINVAL; rc = -EINVAL;
goto init_cc_res_err; goto post_clk_err;
} }
SSI_LOG_DEBUG("CC SIGNATURE=0x%08X\n", signature_val); SSI_LOG_DEBUG("CC SIGNATURE=0x%08X\n", signature_val);
...@@ -306,21 +308,26 @@ static int init_cc_resources(struct platform_device *plat_dev) ...@@ -306,21 +308,26 @@ static int init_cc_resources(struct platform_device *plat_dev)
rc = init_cc_regs(new_drvdata, true); rc = init_cc_regs(new_drvdata, true);
if (unlikely(rc != 0)) { if (unlikely(rc != 0)) {
SSI_LOG_ERR("init_cc_regs failed\n"); SSI_LOG_ERR("init_cc_regs failed\n");
goto init_cc_res_err; goto post_clk_err;
} }
#ifdef ENABLE_CC_SYSFS #ifdef ENABLE_CC_SYSFS
rc = ssi_sysfs_init(&plat_dev->dev.kobj, new_drvdata); rc = ssi_sysfs_init(&plat_dev->dev.kobj, new_drvdata);
if (unlikely(rc != 0)) { if (unlikely(rc != 0)) {
SSI_LOG_ERR("init_stat_db failed\n"); SSI_LOG_ERR("init_stat_db failed\n");
goto init_cc_res_err; goto post_regs_err;
} }
#endif #endif
rc = ssi_fips_init(new_drvdata);
if (unlikely(rc != 0)) {
SSI_LOG_ERR("SSI_FIPS_INIT failed 0x%x\n", rc);
goto post_sysfs_err;
}
rc = ssi_sram_mgr_init(new_drvdata); rc = ssi_sram_mgr_init(new_drvdata);
if (unlikely(rc != 0)) { if (unlikely(rc != 0)) {
SSI_LOG_ERR("ssi_sram_mgr_init failed\n"); SSI_LOG_ERR("ssi_sram_mgr_init failed\n");
goto init_cc_res_err; goto post_fips_init_err;
} }
new_drvdata->mlli_sram_addr = new_drvdata->mlli_sram_addr =
...@@ -328,57 +335,51 @@ static int init_cc_resources(struct platform_device *plat_dev) ...@@ -328,57 +335,51 @@ static int init_cc_resources(struct platform_device *plat_dev)
if (unlikely(new_drvdata->mlli_sram_addr == NULL_SRAM_ADDR)) { if (unlikely(new_drvdata->mlli_sram_addr == NULL_SRAM_ADDR)) {
SSI_LOG_ERR("Failed to alloc MLLI Sram buffer\n"); SSI_LOG_ERR("Failed to alloc MLLI Sram buffer\n");
rc = -ENOMEM; rc = -ENOMEM;
goto init_cc_res_err; goto post_sram_mgr_err;
} }
rc = request_mgr_init(new_drvdata); rc = request_mgr_init(new_drvdata);
if (unlikely(rc != 0)) { if (unlikely(rc != 0)) {
SSI_LOG_ERR("request_mgr_init failed\n"); SSI_LOG_ERR("request_mgr_init failed\n");
goto init_cc_res_err; goto post_sram_mgr_err;
} }
rc = ssi_buffer_mgr_init(new_drvdata); rc = ssi_buffer_mgr_init(new_drvdata);
if (unlikely(rc != 0)) { if (unlikely(rc != 0)) {
SSI_LOG_ERR("buffer_mgr_init failed\n"); SSI_LOG_ERR("buffer_mgr_init failed\n");
goto init_cc_res_err; goto post_req_mgr_err;
} }
rc = ssi_power_mgr_init(new_drvdata); rc = ssi_power_mgr_init(new_drvdata);
if (unlikely(rc != 0)) { if (unlikely(rc != 0)) {
SSI_LOG_ERR("ssi_power_mgr_init failed\n"); SSI_LOG_ERR("ssi_power_mgr_init failed\n");
goto init_cc_res_err; goto post_buf_mgr_err;
}
rc = ssi_fips_init(new_drvdata);
if (unlikely(rc != 0)) {
SSI_LOG_ERR("SSI_FIPS_INIT failed 0x%x\n", rc);
goto init_cc_res_err;
} }
rc = ssi_ivgen_init(new_drvdata); rc = ssi_ivgen_init(new_drvdata);
if (unlikely(rc != 0)) { if (unlikely(rc != 0)) {
SSI_LOG_ERR("ssi_ivgen_init failed\n"); SSI_LOG_ERR("ssi_ivgen_init failed\n");
goto init_cc_res_err; goto post_power_mgr_err;
} }
/* Allocate crypto algs */ /* Allocate crypto algs */
rc = ssi_ablkcipher_alloc(new_drvdata); rc = ssi_ablkcipher_alloc(new_drvdata);
if (unlikely(rc != 0)) { if (unlikely(rc != 0)) {
SSI_LOG_ERR("ssi_ablkcipher_alloc failed\n"); SSI_LOG_ERR("ssi_ablkcipher_alloc failed\n");
goto init_cc_res_err; goto post_ivgen_err;
} }
/* hash must be allocated before aead since hash exports APIs */ /* hash must be allocated before aead since hash exports APIs */
rc = ssi_hash_alloc(new_drvdata); rc = ssi_hash_alloc(new_drvdata);
if (unlikely(rc != 0)) { if (unlikely(rc != 0)) {
SSI_LOG_ERR("ssi_hash_alloc failed\n"); SSI_LOG_ERR("ssi_hash_alloc failed\n");
goto init_cc_res_err; goto post_cipher_err;
} }
rc = ssi_aead_alloc(new_drvdata); rc = ssi_aead_alloc(new_drvdata);
if (unlikely(rc != 0)) { if (unlikely(rc != 0)) {
SSI_LOG_ERR("ssi_aead_alloc failed\n"); SSI_LOG_ERR("ssi_aead_alloc failed\n");
goto init_cc_res_err; goto post_hash_err;
} }
/* If we got here and FIPS mode is enabled /* If we got here and FIPS mode is enabled
...@@ -389,24 +390,33 @@ static int init_cc_resources(struct platform_device *plat_dev) ...@@ -389,24 +390,33 @@ static int init_cc_resources(struct platform_device *plat_dev)
return 0; return 0;
init_cc_res_err: post_hash_err:
SSI_LOG_ERR("Freeing CC HW resources!\n"); ssi_hash_free(new_drvdata);
post_cipher_err:
if (new_drvdata) { ssi_ablkcipher_free(new_drvdata);
ssi_aead_free(new_drvdata); post_ivgen_err:
ssi_hash_free(new_drvdata); ssi_ivgen_fini(new_drvdata);
ssi_ablkcipher_free(new_drvdata); post_power_mgr_err:
ssi_ivgen_fini(new_drvdata); ssi_power_mgr_fini(new_drvdata);
ssi_power_mgr_fini(new_drvdata); post_buf_mgr_err:
ssi_buffer_mgr_fini(new_drvdata); ssi_buffer_mgr_fini(new_drvdata);
request_mgr_fini(new_drvdata); post_req_mgr_err:
ssi_sram_mgr_fini(new_drvdata); request_mgr_fini(new_drvdata);
ssi_fips_fini(new_drvdata); post_sram_mgr_err:
ssi_sram_mgr_fini(new_drvdata);
post_fips_init_err:
ssi_fips_fini(new_drvdata);
post_sysfs_err:
#ifdef ENABLE_CC_SYSFS #ifdef ENABLE_CC_SYSFS
ssi_sysfs_fini(); ssi_sysfs_fini();
#endif #endif
dev_set_drvdata(&plat_dev->dev, NULL); post_regs_err:
} fini_cc_regs(new_drvdata);
post_clk_err:
cc_clk_off(new_drvdata);
post_drvdata_err:
SSI_LOG_ERR("ccree init error occurred!\n");
dev_set_drvdata(&plat_dev->dev, NULL);
return rc; return rc;
} }
......
...@@ -2234,6 +2234,7 @@ int ssi_hash_alloc(struct ssi_drvdata *drvdata) ...@@ -2234,6 +2234,7 @@ int ssi_hash_alloc(struct ssi_drvdata *drvdata)
goto fail; goto fail;
} }
INIT_LIST_HEAD(&hash_handle->hash_list);
drvdata->hash_handle = hash_handle; drvdata->hash_handle = hash_handle;
sram_size_to_alloc = sizeof(digest_len_init) + sram_size_to_alloc = sizeof(digest_len_init) +
...@@ -2264,8 +2265,6 @@ int ssi_hash_alloc(struct ssi_drvdata *drvdata) ...@@ -2264,8 +2265,6 @@ int ssi_hash_alloc(struct ssi_drvdata *drvdata)
goto fail; goto fail;
} }
INIT_LIST_HEAD(&hash_handle->hash_list);
/* ahash registration */ /* ahash registration */
for (alg = 0; alg < ARRAY_SIZE(driver_hash); alg++) { for (alg = 0; alg < ARRAY_SIZE(driver_hash); alg++) {
struct ssi_hash_alg *t_alg; struct ssi_hash_alg *t_alg;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment