162306a36Sopenharmony_ci/* SPDX-License-Identifier: GPL-2.0-only */
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * AMD Secure Processor driver
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (C) 2017-2019 Advanced Micro Devices, Inc.
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Author: Tom Lendacky <thomas.lendacky@amd.com>
862306a36Sopenharmony_ci * Author: Gary R Hook <gary.hook@amd.com>
962306a36Sopenharmony_ci * Author: Brijesh Singh <brijesh.singh@amd.com>
1062306a36Sopenharmony_ci */
1162306a36Sopenharmony_ci
1262306a36Sopenharmony_ci#ifndef __SP_DEV_H__
1362306a36Sopenharmony_ci#define __SP_DEV_H__
1462306a36Sopenharmony_ci
1562306a36Sopenharmony_ci#include <linux/device.h>
1662306a36Sopenharmony_ci#include <linux/spinlock.h>
1762306a36Sopenharmony_ci#include <linux/mutex.h>
1862306a36Sopenharmony_ci#include <linux/list.h>
1962306a36Sopenharmony_ci#include <linux/wait.h>
2062306a36Sopenharmony_ci#include <linux/dmapool.h>
2162306a36Sopenharmony_ci#include <linux/hw_random.h>
2262306a36Sopenharmony_ci#include <linux/bitops.h>
2362306a36Sopenharmony_ci#include <linux/interrupt.h>
2462306a36Sopenharmony_ci#include <linux/irqreturn.h>
2562306a36Sopenharmony_ci
2662306a36Sopenharmony_ci#define SP_MAX_NAME_LEN		32
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_ci#define CACHE_NONE			0x00
2962306a36Sopenharmony_ci#define CACHE_WB_NO_ALLOC		0xb7
3062306a36Sopenharmony_ci
3162306a36Sopenharmony_ci#define PLATFORM_FEATURE_DBC		0x1
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ci#define PSP_FEATURE(psp, feat)	(psp->vdata && psp->vdata->platform_features & PLATFORM_FEATURE_##feat)
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_ci/* Structure to hold CCP device data */
3662306a36Sopenharmony_cistruct ccp_device;
3762306a36Sopenharmony_cistruct ccp_vdata {
3862306a36Sopenharmony_ci	const unsigned int version;
3962306a36Sopenharmony_ci	const unsigned int dma_chan_attr;
4062306a36Sopenharmony_ci	void (*setup)(struct ccp_device *);
4162306a36Sopenharmony_ci	const struct ccp_actions *perform;
4262306a36Sopenharmony_ci	const unsigned int offset;
4362306a36Sopenharmony_ci	const unsigned int rsamax;
4462306a36Sopenharmony_ci};
4562306a36Sopenharmony_ci
4662306a36Sopenharmony_cistruct sev_vdata {
4762306a36Sopenharmony_ci	const unsigned int cmdresp_reg;
4862306a36Sopenharmony_ci	const unsigned int cmdbuff_addr_lo_reg;
4962306a36Sopenharmony_ci	const unsigned int cmdbuff_addr_hi_reg;
5062306a36Sopenharmony_ci};
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_cistruct tee_vdata {
5362306a36Sopenharmony_ci	const unsigned int cmdresp_reg;
5462306a36Sopenharmony_ci	const unsigned int cmdbuff_addr_lo_reg;
5562306a36Sopenharmony_ci	const unsigned int cmdbuff_addr_hi_reg;
5662306a36Sopenharmony_ci	const unsigned int ring_wptr_reg;
5762306a36Sopenharmony_ci	const unsigned int ring_rptr_reg;
5862306a36Sopenharmony_ci	const unsigned int info_reg;
5962306a36Sopenharmony_ci};
6062306a36Sopenharmony_ci
6162306a36Sopenharmony_cistruct platform_access_vdata {
6262306a36Sopenharmony_ci	const unsigned int cmdresp_reg;
6362306a36Sopenharmony_ci	const unsigned int cmdbuff_addr_lo_reg;
6462306a36Sopenharmony_ci	const unsigned int cmdbuff_addr_hi_reg;
6562306a36Sopenharmony_ci	const unsigned int doorbell_button_reg;
6662306a36Sopenharmony_ci	const unsigned int doorbell_cmd_reg;
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_ci};
6962306a36Sopenharmony_ci
7062306a36Sopenharmony_cistruct psp_vdata {
7162306a36Sopenharmony_ci	const struct sev_vdata *sev;
7262306a36Sopenharmony_ci	const struct tee_vdata *tee;
7362306a36Sopenharmony_ci	const struct platform_access_vdata *platform_access;
7462306a36Sopenharmony_ci	const unsigned int feature_reg;
7562306a36Sopenharmony_ci	const unsigned int inten_reg;
7662306a36Sopenharmony_ci	const unsigned int intsts_reg;
7762306a36Sopenharmony_ci	const unsigned int bootloader_info_reg;
7862306a36Sopenharmony_ci	const unsigned int platform_features;
7962306a36Sopenharmony_ci};
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_ci/* Structure to hold SP device data */
8262306a36Sopenharmony_cistruct sp_dev_vdata {
8362306a36Sopenharmony_ci	const unsigned int bar;
8462306a36Sopenharmony_ci
8562306a36Sopenharmony_ci	const struct ccp_vdata *ccp_vdata;
8662306a36Sopenharmony_ci	const struct psp_vdata *psp_vdata;
8762306a36Sopenharmony_ci};
8862306a36Sopenharmony_ci
8962306a36Sopenharmony_cistruct sp_device {
9062306a36Sopenharmony_ci	struct list_head entry;
9162306a36Sopenharmony_ci
9262306a36Sopenharmony_ci	struct device *dev;
9362306a36Sopenharmony_ci
9462306a36Sopenharmony_ci	struct sp_dev_vdata *dev_vdata;
9562306a36Sopenharmony_ci	unsigned int ord;
9662306a36Sopenharmony_ci	char name[SP_MAX_NAME_LEN];
9762306a36Sopenharmony_ci
9862306a36Sopenharmony_ci	/* Bus specific device information */
9962306a36Sopenharmony_ci	void *dev_specific;
10062306a36Sopenharmony_ci
10162306a36Sopenharmony_ci	/* I/O area used for device communication. */
10262306a36Sopenharmony_ci	void __iomem *io_map;
10362306a36Sopenharmony_ci
10462306a36Sopenharmony_ci	/* DMA caching attribute support */
10562306a36Sopenharmony_ci	unsigned int axcache;
10662306a36Sopenharmony_ci
10762306a36Sopenharmony_ci	/* get and set master device */
10862306a36Sopenharmony_ci	struct sp_device*(*get_psp_master_device)(void);
10962306a36Sopenharmony_ci	void (*set_psp_master_device)(struct sp_device *);
11062306a36Sopenharmony_ci	void (*clear_psp_master_device)(struct sp_device *);
11162306a36Sopenharmony_ci
11262306a36Sopenharmony_ci	bool irq_registered;
11362306a36Sopenharmony_ci	bool use_tasklet;
11462306a36Sopenharmony_ci
11562306a36Sopenharmony_ci	unsigned int ccp_irq;
11662306a36Sopenharmony_ci	irq_handler_t ccp_irq_handler;
11762306a36Sopenharmony_ci	void *ccp_irq_data;
11862306a36Sopenharmony_ci
11962306a36Sopenharmony_ci	unsigned int psp_irq;
12062306a36Sopenharmony_ci	irq_handler_t psp_irq_handler;
12162306a36Sopenharmony_ci	void *psp_irq_data;
12262306a36Sopenharmony_ci
12362306a36Sopenharmony_ci	void *ccp_data;
12462306a36Sopenharmony_ci	void *psp_data;
12562306a36Sopenharmony_ci};
12662306a36Sopenharmony_ci
12762306a36Sopenharmony_ciint sp_pci_init(void);
12862306a36Sopenharmony_civoid sp_pci_exit(void);
12962306a36Sopenharmony_ci
13062306a36Sopenharmony_ciint sp_platform_init(void);
13162306a36Sopenharmony_civoid sp_platform_exit(void);
13262306a36Sopenharmony_ci
13362306a36Sopenharmony_cistruct sp_device *sp_alloc_struct(struct device *dev);
13462306a36Sopenharmony_ci
13562306a36Sopenharmony_ciint sp_init(struct sp_device *sp);
13662306a36Sopenharmony_civoid sp_destroy(struct sp_device *sp);
13762306a36Sopenharmony_cistruct sp_device *sp_get_master(void);
13862306a36Sopenharmony_ci
13962306a36Sopenharmony_ciint sp_suspend(struct sp_device *sp);
14062306a36Sopenharmony_ciint sp_resume(struct sp_device *sp);
14162306a36Sopenharmony_ciint sp_request_ccp_irq(struct sp_device *sp, irq_handler_t handler,
14262306a36Sopenharmony_ci		       const char *name, void *data);
14362306a36Sopenharmony_civoid sp_free_ccp_irq(struct sp_device *sp, void *data);
14462306a36Sopenharmony_ciint sp_request_psp_irq(struct sp_device *sp, irq_handler_t handler,
14562306a36Sopenharmony_ci		       const char *name, void *data);
14662306a36Sopenharmony_civoid sp_free_psp_irq(struct sp_device *sp, void *data);
14762306a36Sopenharmony_cistruct sp_device *sp_get_psp_master_device(void);
14862306a36Sopenharmony_ci
14962306a36Sopenharmony_ci#ifdef CONFIG_CRYPTO_DEV_SP_CCP
15062306a36Sopenharmony_ci
15162306a36Sopenharmony_ciint ccp_dev_init(struct sp_device *sp);
15262306a36Sopenharmony_civoid ccp_dev_destroy(struct sp_device *sp);
15362306a36Sopenharmony_ci
15462306a36Sopenharmony_civoid ccp_dev_suspend(struct sp_device *sp);
15562306a36Sopenharmony_civoid ccp_dev_resume(struct sp_device *sp);
15662306a36Sopenharmony_ci
15762306a36Sopenharmony_ci#else	/* !CONFIG_CRYPTO_DEV_SP_CCP */
15862306a36Sopenharmony_ci
15962306a36Sopenharmony_cistatic inline int ccp_dev_init(struct sp_device *sp)
16062306a36Sopenharmony_ci{
16162306a36Sopenharmony_ci	return 0;
16262306a36Sopenharmony_ci}
16362306a36Sopenharmony_cistatic inline void ccp_dev_destroy(struct sp_device *sp) { }
16462306a36Sopenharmony_cistatic inline void ccp_dev_suspend(struct sp_device *sp) { }
16562306a36Sopenharmony_cistatic inline void ccp_dev_resume(struct sp_device *sp) { }
16662306a36Sopenharmony_ci#endif	/* CONFIG_CRYPTO_DEV_SP_CCP */
16762306a36Sopenharmony_ci
16862306a36Sopenharmony_ci#ifdef CONFIG_CRYPTO_DEV_SP_PSP
16962306a36Sopenharmony_ci
17062306a36Sopenharmony_ciint psp_dev_init(struct sp_device *sp);
17162306a36Sopenharmony_civoid psp_pci_init(void);
17262306a36Sopenharmony_civoid psp_dev_destroy(struct sp_device *sp);
17362306a36Sopenharmony_civoid psp_pci_exit(void);
17462306a36Sopenharmony_ci
17562306a36Sopenharmony_ci#else /* !CONFIG_CRYPTO_DEV_SP_PSP */
17662306a36Sopenharmony_ci
17762306a36Sopenharmony_cistatic inline int psp_dev_init(struct sp_device *sp) { return 0; }
17862306a36Sopenharmony_cistatic inline void psp_pci_init(void) { }
17962306a36Sopenharmony_cistatic inline void psp_dev_destroy(struct sp_device *sp) { }
18062306a36Sopenharmony_cistatic inline void psp_pci_exit(void) { }
18162306a36Sopenharmony_ci
18262306a36Sopenharmony_ci#endif /* CONFIG_CRYPTO_DEV_SP_PSP */
18362306a36Sopenharmony_ci
18462306a36Sopenharmony_ci#endif
185