blob: 5a01bee050e7dd0a5696e1670f803f1f76ca268c [file] [log] [blame]
Olav Haugan99660ca2012-12-04 13:30:41 -08001/* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12#include <linux/err.h>
13#include <linux/mutex.h>
14#include <linux/list.h>
15
16#ifndef MSM_IOMMU_PERFMON_H
17#define MSM_IOMMU_PERFMON_H
18
Olav Haugan99660ca2012-12-04 13:30:41 -080019/**
20 * struct iommu_access_ops - Callbacks for accessing IOMMU
21 * @iommu_power_on: Turn on clocks/power to unit
22 * @iommu_power_off: Turn off clocks/power to unit
23 * @iommu_lock_acquire: Acquire any locks needed
24 * @iommu_lock_release: Release locks needed
25 */
26struct iommu_access_ops {
27 int (*iommu_power_on)(void *);
28 int (*iommu_power_off)(void *);
29 void (*iommu_lock_acquire)(void);
30 void (*iommu_lock_release)(void);
31};
32
33/**
34 * struct iommu_pmon_counter - container for a performance counter.
35 * @counter_no: counter number within the group
36 * @absolute_counter_no: counter number within IOMMU PMU
37 * @value: cached counter value
38 * @overflow_count: no of times counter has overflowed
39 * @enabled: indicates whether counter is enabled or not
40 * @current_event_class: current selected event class, -1 if none
41 * @counter_dir: debugfs directory for this counter
42 * @cnt_group: group this counter belongs to
43 */
44struct iommu_pmon_counter {
45 unsigned int counter_no;
46 unsigned int absolute_counter_no;
47 unsigned long value;
48 unsigned long overflow_count;
49 unsigned int enabled;
50 int current_event_class;
51 struct dentry *counter_dir;
52 struct iommu_pmon_cnt_group *cnt_group;
53};
54
55/**
56 * struct iommu_pmon_cnt_group - container for a perf mon counter group.
57 * @grp_no: group number
58 * @num_counters: number of counters in this group
59 * @counters: list of counter in this group
60 * @group_dir: debugfs directory for this group
61 * @pmon: pointer to the iommu_pmon object this group belongs to
62 */
63struct iommu_pmon_cnt_group {
64 unsigned int grp_no;
65 unsigned int num_counters;
66 struct iommu_pmon_counter *counters;
67 struct dentry *group_dir;
68 struct iommu_pmon *pmon;
69};
70
71/**
72 * struct iommu_info - container for a perf mon iommu info.
73 * @iommu_name: name of the iommu from device tree
74 * @base: virtual base address for this iommu
75 * @evt_irq: irq number for event overflow interrupt
76 * @iommu_dev: pointer to iommu device
77 * @ops: iommu access operations pointer.
Olav Hauganef69e892013-02-04 13:47:08 -080078 * @hw_ops: iommu pm hw access operations pointer.
Olav Haugan99660ca2012-12-04 13:30:41 -080079 */
80struct iommu_info {
81 const char *iommu_name;
82 void *base;
83 int evt_irq;
84 struct device *iommu_dev;
85 struct iommu_access_ops *ops;
Olav Hauganef69e892013-02-04 13:47:08 -080086 struct iommu_pm_hw_ops *hw_ops;
Olav Haugan99660ca2012-12-04 13:30:41 -080087};
88
89/**
90 * struct iommu_pmon - main container for a perf mon data.
91 * @iommu_dir: debugfs directory for this iommu
92 * @iommu: iommu_info instance
93 * @iommu_list: iommu_list head
94 * @cnt_grp: list of counter groups
95 * @num_groups: number of counter groups
Olav Haugan0c2d9322013-01-31 18:35:30 -080096 * @num_counters: number of counters per group
97 * @event_cls_supported: an array of event classes supported for this PMU
98 * @nevent_cls_supported: number of event classes supported.
Olav Haugan99660ca2012-12-04 13:30:41 -080099 * @enabled: Indicates whether perf. mon is enabled or not
100 * @iommu_attached Indicates whether iommu is attached or not.
101 * @lock: mutex used to synchronize access to shared data
102 */
103struct iommu_pmon {
104 struct dentry *iommu_dir;
105 struct iommu_info iommu;
106 struct list_head iommu_list;
107 struct iommu_pmon_cnt_group *cnt_grp;
Olav Haugan0c2d9322013-01-31 18:35:30 -0800108 u32 num_groups;
109 u32 num_counters;
110 u32 *event_cls_supported;
111 u32 nevent_cls_supported;
Olav Haugan99660ca2012-12-04 13:30:41 -0800112 unsigned int enabled;
113 unsigned int iommu_attach_count;
114 struct mutex lock;
115};
116
Olav Hauganef69e892013-02-04 13:47:08 -0800117/**
118 * struct iommu_hw_ops - Callbacks for accessing IOMMU HW
Olav Haugan7a2f99c2013-02-04 14:43:26 -0800119 * @initialize_hw: Call to do any initialization before enabling ovf interrupts
Olav Hauganef69e892013-02-04 13:47:08 -0800120 * @is_hw_access_ok: Returns 1 if we can access HW, 0 otherwise
121 * @grp_enable: Call to enable a counter group
122 * @grp_disable: Call to disable a counter group
123 * @enable_pm: Call to enable PM
124 * @disable_pm: Call to disable PM
125 * @reset_counters: Call to reset counters
126 * @check_for_overflow: Call to check for overflow
127 * @evt_ovfl_int_handler: Overflow interrupt handler callback
128 * @counter_enable: Call to enable counters
129 * @counter_disable: Call to disable counters
130 * @ovfl_int_enable: Call to enable overflow interrupts
131 * @ovfl_int_disable: Call to disable overflow interrupts
132 * @set_event_class: Call to set event class
133 * @read_counter: Call to read a counter value
134 */
135struct iommu_pm_hw_ops {
Olav Haugan7a2f99c2013-02-04 14:43:26 -0800136 void (*initialize_hw)(const struct iommu_pmon *);
Olav Hauganef69e892013-02-04 13:47:08 -0800137 unsigned int (*is_hw_access_OK)(const struct iommu_pmon *);
138 void (*grp_enable)(struct iommu_info *, unsigned int);
139 void (*grp_disable)(struct iommu_info *, unsigned int);
140 void (*enable_pm)(struct iommu_info *);
141 void (*disable_pm)(struct iommu_info *);
142 void (*reset_counters)(const struct iommu_info *);
143 void (*check_for_overflow)(struct iommu_pmon *);
144 irqreturn_t (*evt_ovfl_int_handler)(int, void *);
145 void (*counter_enable)(struct iommu_info *,
146 struct iommu_pmon_counter *);
147 void (*counter_disable)(struct iommu_info *,
148 struct iommu_pmon_counter *);
149 void (*ovfl_int_enable)(struct iommu_info *,
150 const struct iommu_pmon_counter *);
151 void (*ovfl_int_disable)(struct iommu_info *,
152 const struct iommu_pmon_counter *);
153 void (*set_event_class)(struct iommu_pmon *pmon, unsigned int,
154 unsigned int);
155 unsigned int (*read_counter)(struct iommu_pmon_counter *);
156};
157
Olav Haugan7a2f99c2013-02-04 14:43:26 -0800158extern struct iommu_access_ops iommu_access_ops_v0;
Olav Hauganef69e892013-02-04 13:47:08 -0800159extern struct iommu_access_ops iommu_access_ops_v1;
Olav Haugan7a2f99c2013-02-04 14:43:26 -0800160#define MSM_IOMMU_PMU_NO_EVENT_CLASS -1
Olav Haugan99660ca2012-12-04 13:30:41 -0800161
162#ifdef CONFIG_MSM_IOMMU_PMON
Olav Hauganef69e892013-02-04 13:47:08 -0800163
164/**
Olav Haugan7a2f99c2013-02-04 14:43:26 -0800165 * Get pointer to PMU hardware access functions for IOMMUv0 PMU
166 */
167struct iommu_pm_hw_ops *iommu_pm_get_hw_ops_v0(void);
168
169/**
Olav Hauganef69e892013-02-04 13:47:08 -0800170 * Get pointer to PMU hardware access functions for IOMMUv1 PMU
171 */
172struct iommu_pm_hw_ops *iommu_pm_get_hw_ops_v1(void);
173
Olav Haugan99660ca2012-12-04 13:30:41 -0800174/**
175 * Allocate memory for performance monitor structure. Must
Olav Haugan0c2d9322013-01-31 18:35:30 -0800176 * be called before iommu_pm_iommu_register
Olav Haugan99660ca2012-12-04 13:30:41 -0800177 */
Olav Haugan0c2d9322013-01-31 18:35:30 -0800178struct iommu_pmon *msm_iommu_pm_alloc(struct device *iommu_dev);
Olav Haugan99660ca2012-12-04 13:30:41 -0800179
180/**
181 * Free memory previously allocated with iommu_pm_alloc
182 */
183void msm_iommu_pm_free(struct device *iommu_dev);
184
185/**
186 * Register iommu with the performance monitor module.
187 */
Olav Haugan0c2d9322013-01-31 18:35:30 -0800188int msm_iommu_pm_iommu_register(struct iommu_pmon *info);
Olav Haugan99660ca2012-12-04 13:30:41 -0800189
190/**
191 * Unregister iommu with the performance monitor module.
192 */
193void msm_iommu_pm_iommu_unregister(struct device *dev);
194
195/**
196 * Called by iommu driver when attaching is complete
197 * Must NOT be called with IOMMU mutexes held.
198 * @param iommu_dev IOMMU device that is attached
199 */
200void msm_iommu_attached(struct device *dev);
201
202/**
203 * Called by iommu driver before detaching.
204 * Must NOT be called with IOMMU mutexes held.
205 * @param iommu_dev IOMMU device that is going to be detached
206 */
207void msm_iommu_detached(struct device *dev);
208#else
Olav Haugan7a2f99c2013-02-04 14:43:26 -0800209static inline struct iommu_pm_hw_ops *iommu_pm_get_hw_ops_v0(void)
210{
211 return NULL;
212}
213
Olav Hauganef69e892013-02-04 13:47:08 -0800214static inline struct iommu_pm_hw_ops *iommu_pm_get_hw_ops_v1(void)
215{
216 return NULL;
217}
218
Olav Haugan0c2d9322013-01-31 18:35:30 -0800219static inline struct iommu_pmon *msm_iommu_pm_alloc(struct device *iommu_dev)
Olav Haugan99660ca2012-12-04 13:30:41 -0800220{
221 return NULL;
222}
223
224static inline void msm_iommu_pm_free(struct device *iommu_dev)
225{
226 return;
227}
228
Olav Haugan0c2d9322013-01-31 18:35:30 -0800229static inline int msm_iommu_pm_iommu_register(struct iommu_pmon *info)
Olav Haugan99660ca2012-12-04 13:30:41 -0800230{
231 return -EIO;
232}
233
234static inline void msm_iommu_pm_iommu_unregister(struct device *dev)
235{
236}
237
238static inline void msm_iommu_attached(struct device *dev)
239{
240}
241
242static inline void msm_iommu_detached(struct device *dev)
243{
244}
245#endif
246#endif