|  | /* | 
|  | * Keystone Navigator QMSS driver internal header | 
|  | * | 
|  | * Copyright (C) 2014 Texas Instruments Incorporated - http://www.ti.com | 
|  | * Author:	Sandeep Nair <sandeep_n@ti.com> | 
|  | *		Cyril Chemparathy <cyril@ti.com> | 
|  | *		Santosh Shilimkar <santosh.shilimkar@ti.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. | 
|  | * | 
|  | * This program is distributed in the hope that it will be useful, but | 
|  | * WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
|  | * General Public License for more details. | 
|  | */ | 
|  |  | 
|  | #ifndef __KNAV_QMSS_H__ | 
|  | #define __KNAV_QMSS_H__ | 
|  |  | 
|  | #define THRESH_GTE	BIT(7) | 
|  | #define THRESH_LT	0 | 
|  |  | 
|  | #define PDSP_CTRL_PC_MASK	0xffff0000 | 
|  | #define PDSP_CTRL_SOFT_RESET	BIT(0) | 
|  | #define PDSP_CTRL_ENABLE	BIT(1) | 
|  | #define PDSP_CTRL_RUNNING	BIT(15) | 
|  |  | 
|  | #define ACC_MAX_CHANNEL		48 | 
|  | #define ACC_DEFAULT_PERIOD	25 /* usecs */ | 
|  |  | 
|  | #define ACC_CHANNEL_INT_BASE		2 | 
|  |  | 
|  | #define ACC_LIST_ENTRY_TYPE		1 | 
|  | #define ACC_LIST_ENTRY_WORDS		(1 << ACC_LIST_ENTRY_TYPE) | 
|  | #define ACC_LIST_ENTRY_QUEUE_IDX	0 | 
|  | #define ACC_LIST_ENTRY_DESC_IDX	(ACC_LIST_ENTRY_WORDS - 1) | 
|  |  | 
|  | #define ACC_CMD_DISABLE_CHANNEL	0x80 | 
|  | #define ACC_CMD_ENABLE_CHANNEL	0x81 | 
|  | #define ACC_CFG_MULTI_QUEUE		BIT(21) | 
|  |  | 
|  | #define ACC_INTD_OFFSET_EOI		(0x0010) | 
|  | #define ACC_INTD_OFFSET_COUNT(ch)	(0x0300 + 4 * (ch)) | 
|  | #define ACC_INTD_OFFSET_STATUS(ch)	(0x0200 + 4 * ((ch) / 32)) | 
|  |  | 
|  | #define RANGE_MAX_IRQS			64 | 
|  |  | 
|  | #define ACC_DESCS_MAX		SZ_1K | 
|  | #define ACC_DESCS_MASK		(ACC_DESCS_MAX - 1) | 
|  | #define DESC_SIZE_MASK		0xful | 
|  | #define DESC_PTR_MASK		(~DESC_SIZE_MASK) | 
|  |  | 
|  | #define KNAV_NAME_SIZE			32 | 
|  |  | 
|  | enum knav_acc_result { | 
|  | ACC_RET_IDLE, | 
|  | ACC_RET_SUCCESS, | 
|  | ACC_RET_INVALID_COMMAND, | 
|  | ACC_RET_INVALID_CHANNEL, | 
|  | ACC_RET_INACTIVE_CHANNEL, | 
|  | ACC_RET_ACTIVE_CHANNEL, | 
|  | ACC_RET_INVALID_QUEUE, | 
|  | ACC_RET_INVALID_RET, | 
|  | }; | 
|  |  | 
|  | struct knav_reg_config { | 
|  | u32		revision; | 
|  | u32		__pad1; | 
|  | u32		divert; | 
|  | u32		link_ram_base0; | 
|  | u32		link_ram_size0; | 
|  | u32		link_ram_base1; | 
|  | u32		__pad2[2]; | 
|  | u32		starvation[0]; | 
|  | }; | 
|  |  | 
|  | struct knav_reg_region { | 
|  | u32		base; | 
|  | u32		start_index; | 
|  | u32		size_count; | 
|  | u32		__pad; | 
|  | }; | 
|  |  | 
|  | struct knav_reg_pdsp_regs { | 
|  | u32		control; | 
|  | u32		status; | 
|  | u32		cycle_count; | 
|  | u32		stall_count; | 
|  | }; | 
|  |  | 
|  | struct knav_reg_acc_command { | 
|  | u32		command; | 
|  | u32		queue_mask; | 
|  | u32		list_phys; | 
|  | u32		queue_num; | 
|  | u32		timer_config; | 
|  | }; | 
|  |  | 
|  | struct knav_link_ram_block { | 
|  | dma_addr_t	 phys; | 
|  | void		*virt; | 
|  | size_t		 size; | 
|  | }; | 
|  |  | 
|  | struct knav_acc_info { | 
|  | u32			 pdsp_id; | 
|  | u32			 start_channel; | 
|  | u32			 list_entries; | 
|  | u32			 pacing_mode; | 
|  | u32			 timer_count; | 
|  | int			 mem_size; | 
|  | int			 list_size; | 
|  | struct knav_pdsp_info	*pdsp; | 
|  | }; | 
|  |  | 
|  | struct knav_acc_channel { | 
|  | u32			channel; | 
|  | u32			list_index; | 
|  | u32			open_mask; | 
|  | u32			*list_cpu[2]; | 
|  | dma_addr_t		list_dma[2]; | 
|  | char			name[KNAV_NAME_SIZE]; | 
|  | atomic_t		retrigger_count; | 
|  | }; | 
|  |  | 
|  | struct knav_pdsp_info { | 
|  | const char					*name; | 
|  | struct knav_reg_pdsp_regs  __iomem		*regs; | 
|  | union { | 
|  | void __iomem				*command; | 
|  | struct knav_reg_acc_command __iomem	*acc_command; | 
|  | u32 __iomem				*qos_command; | 
|  | }; | 
|  | void __iomem					*intd; | 
|  | u32 __iomem					*iram; | 
|  | const char					*firmware; | 
|  | u32						id; | 
|  | struct list_head				list; | 
|  | }; | 
|  |  | 
|  | struct knav_qmgr_info { | 
|  | unsigned			start_queue; | 
|  | unsigned			num_queues; | 
|  | struct knav_reg_config __iomem	*reg_config; | 
|  | struct knav_reg_region __iomem	*reg_region; | 
|  | struct knav_reg_queue __iomem	*reg_push, *reg_pop, *reg_peek; | 
|  | void __iomem			*reg_status; | 
|  | struct list_head		list; | 
|  | }; | 
|  |  | 
|  | #define KNAV_NUM_LINKRAM	2 | 
|  |  | 
|  | /** | 
|  | * struct knav_queue_stats:	queue statistics | 
|  | * pushes:			number of push operations | 
|  | * pops:			number of pop operations | 
|  | * push_errors:			number of push errors | 
|  | * pop_errors:			number of pop errors | 
|  | * notifies:			notifier counts | 
|  | */ | 
|  | struct knav_queue_stats { | 
|  | atomic_t	 pushes; | 
|  | atomic_t	 pops; | 
|  | atomic_t	 push_errors; | 
|  | atomic_t	 pop_errors; | 
|  | atomic_t	 notifies; | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * struct knav_reg_queue:	queue registers | 
|  | * @entry_count:		valid entries in the queue | 
|  | * @byte_count:			total byte count in thhe queue | 
|  | * @packet_size:		packet size for the queue | 
|  | * @ptr_size_thresh:		packet pointer size threshold | 
|  | */ | 
|  | struct knav_reg_queue { | 
|  | u32		entry_count; | 
|  | u32		byte_count; | 
|  | u32		packet_size; | 
|  | u32		ptr_size_thresh; | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * struct knav_region:		qmss region info | 
|  | * @dma_start, dma_end:		start and end dma address | 
|  | * @virt_start, virt_end:	start and end virtual address | 
|  | * @desc_size:			descriptor size | 
|  | * @used_desc:			consumed descriptors | 
|  | * @id:				region number | 
|  | * @num_desc:			total descriptors | 
|  | * @link_index:			index of the first descriptor | 
|  | * @name:			region name | 
|  | * @list:			instance in the device's region list | 
|  | * @pools:			list of descriptor pools in the region | 
|  | */ | 
|  | struct knav_region { | 
|  | dma_addr_t		dma_start, dma_end; | 
|  | void			*virt_start, *virt_end; | 
|  | unsigned		desc_size; | 
|  | unsigned		used_desc; | 
|  | unsigned		id; | 
|  | unsigned		num_desc; | 
|  | unsigned		link_index; | 
|  | const char		*name; | 
|  | struct list_head	list; | 
|  | struct list_head	pools; | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * struct knav_pool:		qmss pools | 
|  | * @dev:			device pointer | 
|  | * @region:			qmss region info | 
|  | * @queue:			queue registers | 
|  | * @kdev:			qmss device pointer | 
|  | * @region_offset:		offset from the base | 
|  | * @num_desc:			total descriptors | 
|  | * @desc_size:			descriptor size | 
|  | * @region_id:			region number | 
|  | * @name:			pool name | 
|  | * @list:			list head | 
|  | * @region_inst:		instance in the region's pool list | 
|  | */ | 
|  | struct knav_pool { | 
|  | struct device			*dev; | 
|  | struct knav_region		*region; | 
|  | struct knav_queue		*queue; | 
|  | struct knav_device		*kdev; | 
|  | int				region_offset; | 
|  | int				num_desc; | 
|  | int				desc_size; | 
|  | int				region_id; | 
|  | const char			*name; | 
|  | struct list_head		list; | 
|  | struct list_head		region_inst; | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * struct knav_queue_inst:		qmss queue instace properties | 
|  | * @descs:				descriptor pointer | 
|  | * @desc_head, desc_tail, desc_count:	descriptor counters | 
|  | * @acc:				accumulator channel pointer | 
|  | * @kdev:				qmss device pointer | 
|  | * @range:				range info | 
|  | * @qmgr:				queue manager info | 
|  | * @id:					queue instace id | 
|  | * @irq_num:				irq line number | 
|  | * @notify_needed:			notifier needed based on queue type | 
|  | * @num_notifiers:			total notifiers | 
|  | * @handles:				list head | 
|  | * @name:				queue instance name | 
|  | * @irq_name:				irq line name | 
|  | */ | 
|  | struct knav_queue_inst { | 
|  | u32				*descs; | 
|  | atomic_t			desc_head, desc_tail, desc_count; | 
|  | struct knav_acc_channel	*acc; | 
|  | struct knav_device		*kdev; | 
|  | struct knav_range_info		*range; | 
|  | struct knav_qmgr_info		*qmgr; | 
|  | u32				id; | 
|  | int				irq_num; | 
|  | int				notify_needed; | 
|  | atomic_t			num_notifiers; | 
|  | struct list_head		handles; | 
|  | const char			*name; | 
|  | const char			*irq_name; | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * struct knav_queue:			qmss queue properties | 
|  | * @reg_push, reg_pop, reg_peek:	push, pop queue registers | 
|  | * @inst:				qmss queue instace properties | 
|  | * @notifier_fn:			notifier function | 
|  | * @notifier_fn_arg:			notifier function argument | 
|  | * @notifier_enabled:			notier enabled for a give queue | 
|  | * @rcu:				rcu head | 
|  | * @flags:				queue flags | 
|  | * @list:				list head | 
|  | */ | 
|  | struct knav_queue { | 
|  | struct knav_reg_queue __iomem	*reg_push, *reg_pop, *reg_peek; | 
|  | struct knav_queue_inst		*inst; | 
|  | struct knav_queue_stats	stats; | 
|  | knav_queue_notify_fn		notifier_fn; | 
|  | void				*notifier_fn_arg; | 
|  | atomic_t			notifier_enabled; | 
|  | struct rcu_head			rcu; | 
|  | unsigned			flags; | 
|  | struct list_head		list; | 
|  | }; | 
|  |  | 
|  | struct knav_device { | 
|  | struct device				*dev; | 
|  | unsigned				base_id; | 
|  | unsigned				num_queues; | 
|  | unsigned				num_queues_in_use; | 
|  | unsigned				inst_shift; | 
|  | struct knav_link_ram_block		link_rams[KNAV_NUM_LINKRAM]; | 
|  | void					*instances; | 
|  | struct list_head			regions; | 
|  | struct list_head			queue_ranges; | 
|  | struct list_head			pools; | 
|  | struct list_head			pdsps; | 
|  | struct list_head			qmgrs; | 
|  | }; | 
|  |  | 
|  | struct knav_range_ops { | 
|  | int	(*init_range)(struct knav_range_info *range); | 
|  | int	(*free_range)(struct knav_range_info *range); | 
|  | int	(*init_queue)(struct knav_range_info *range, | 
|  | struct knav_queue_inst *inst); | 
|  | int	(*open_queue)(struct knav_range_info *range, | 
|  | struct knav_queue_inst *inst, unsigned flags); | 
|  | int	(*close_queue)(struct knav_range_info *range, | 
|  | struct knav_queue_inst *inst); | 
|  | int	(*set_notify)(struct knav_range_info *range, | 
|  | struct knav_queue_inst *inst, bool enabled); | 
|  | }; | 
|  |  | 
|  | struct knav_irq_info { | 
|  | int	irq; | 
|  | u32	cpu_map; | 
|  | }; | 
|  |  | 
|  | struct knav_range_info { | 
|  | const char			*name; | 
|  | struct knav_device		*kdev; | 
|  | unsigned			queue_base; | 
|  | unsigned			num_queues; | 
|  | void				*queue_base_inst; | 
|  | unsigned			flags; | 
|  | struct list_head		list; | 
|  | struct knav_range_ops		*ops; | 
|  | struct knav_acc_info		acc_info; | 
|  | struct knav_acc_channel	*acc; | 
|  | unsigned			num_irqs; | 
|  | struct knav_irq_info		irqs[RANGE_MAX_IRQS]; | 
|  | }; | 
|  |  | 
|  | #define RANGE_RESERVED		BIT(0) | 
|  | #define RANGE_HAS_IRQ		BIT(1) | 
|  | #define RANGE_HAS_ACCUMULATOR	BIT(2) | 
|  | #define RANGE_MULTI_QUEUE	BIT(3) | 
|  |  | 
|  | #define for_each_region(kdev, region)				\ | 
|  | list_for_each_entry(region, &kdev->regions, list) | 
|  |  | 
|  | #define first_region(kdev)					\ | 
|  | list_first_entry_or_null(&kdev->regions, \ | 
|  | struct knav_region, list) | 
|  |  | 
|  | #define for_each_queue_range(kdev, range)			\ | 
|  | list_for_each_entry(range, &kdev->queue_ranges, list) | 
|  |  | 
|  | #define first_queue_range(kdev)					\ | 
|  | list_first_entry_or_null(&kdev->queue_ranges, \ | 
|  | struct knav_range_info, list) | 
|  |  | 
|  | #define for_each_pool(kdev, pool)				\ | 
|  | list_for_each_entry(pool, &kdev->pools, list) | 
|  |  | 
|  | #define for_each_pdsp(kdev, pdsp)				\ | 
|  | list_for_each_entry(pdsp, &kdev->pdsps, list) | 
|  |  | 
|  | #define for_each_qmgr(kdev, qmgr)				\ | 
|  | list_for_each_entry(qmgr, &kdev->qmgrs, list) | 
|  |  | 
|  | static inline struct knav_pdsp_info * | 
|  | knav_find_pdsp(struct knav_device *kdev, unsigned pdsp_id) | 
|  | { | 
|  | struct knav_pdsp_info *pdsp; | 
|  |  | 
|  | for_each_pdsp(kdev, pdsp) | 
|  | if (pdsp_id == pdsp->id) | 
|  | return pdsp; | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | extern int knav_init_acc_range(struct knav_device *kdev, | 
|  | struct device_node *node, | 
|  | struct knav_range_info *range); | 
|  | extern void knav_queue_notify(struct knav_queue_inst *inst); | 
|  |  | 
|  | #endif /* __KNAV_QMSS_H__ */ |