Loading...
1// SPDX-License-Identifier: GPL-2.0
2
3/* Copyright (c) 2012-2018, The Linux Foundation. All rights reserved.
4 * Copyright (C) 2018-2022 Linaro Ltd.
5 */
6
7#include <linux/types.h>
8#include <linux/kernel.h>
9
10#include "ipa.h"
11#include "ipa_data.h"
12#include "ipa_reg.h"
13#include "ipa_resource.h"
14
15/**
16 * DOC: IPA Resources
17 *
18 * The IPA manages a set of resources internally for various purposes.
19 * A given IPA version has a fixed number of resource types, and a fixed
20 * total number of resources of each type. "Source" resource types
21 * are separate from "destination" resource types.
22 *
23 * Each version of IPA also has some number of resource groups. Each
24 * endpoint is assigned to a resource group, and all endpoints in the
25 * same group share pools of each type of resource. A subset of the
26 * total resources of each type is assigned for use by each group.
27 */
28
29static bool ipa_resource_limits_valid(struct ipa *ipa,
30 const struct ipa_resource_data *data)
31{
32 u32 group_count;
33 u32 i;
34 u32 j;
35
36 /* We program at most 8 source or destination resource group limits */
37 BUILD_BUG_ON(IPA_RESOURCE_GROUP_MAX > 8);
38
39 group_count = data->rsrc_group_src_count;
40 if (!group_count || group_count > IPA_RESOURCE_GROUP_MAX)
41 return false;
42
43 /* Return an error if a non-zero resource limit is specified
44 * for a resource group not supported by hardware.
45 */
46 for (i = 0; i < data->resource_src_count; i++) {
47 const struct ipa_resource *resource;
48
49 resource = &data->resource_src[i];
50 for (j = group_count; j < IPA_RESOURCE_GROUP_MAX; j++)
51 if (resource->limits[j].min || resource->limits[j].max)
52 return false;
53 }
54
55 group_count = data->rsrc_group_dst_count;
56 if (!group_count || group_count > IPA_RESOURCE_GROUP_MAX)
57 return false;
58
59 for (i = 0; i < data->resource_dst_count; i++) {
60 const struct ipa_resource *resource;
61
62 resource = &data->resource_dst[i];
63 for (j = group_count; j < IPA_RESOURCE_GROUP_MAX; j++)
64 if (resource->limits[j].min || resource->limits[j].max)
65 return false;
66 }
67
68 return true;
69}
70
71static void
72ipa_resource_config_common(struct ipa *ipa, u32 resource_type,
73 const struct reg *reg,
74 const struct ipa_resource_limits *xlimits,
75 const struct ipa_resource_limits *ylimits)
76{
77 u32 val;
78
79 val = reg_encode(reg, X_MIN_LIM, xlimits->min);
80 val |= reg_encode(reg, X_MAX_LIM, xlimits->max);
81 if (ylimits) {
82 val |= reg_encode(reg, Y_MIN_LIM, ylimits->min);
83 val |= reg_encode(reg, Y_MAX_LIM, ylimits->max);
84 }
85
86 iowrite32(val, ipa->reg_virt + reg_n_offset(reg, resource_type));
87}
88
89static void ipa_resource_config_src(struct ipa *ipa, u32 resource_type,
90 const struct ipa_resource_data *data)
91{
92 u32 group_count = data->rsrc_group_src_count;
93 const struct ipa_resource_limits *ylimits;
94 const struct ipa_resource *resource;
95 const struct reg *reg;
96
97 resource = &data->resource_src[resource_type];
98
99 reg = ipa_reg(ipa, SRC_RSRC_GRP_01_RSRC_TYPE);
100 ylimits = group_count == 1 ? NULL : &resource->limits[1];
101 ipa_resource_config_common(ipa, resource_type, reg,
102 &resource->limits[0], ylimits);
103 if (group_count < 3)
104 return;
105
106 reg = ipa_reg(ipa, SRC_RSRC_GRP_23_RSRC_TYPE);
107 ylimits = group_count == 3 ? NULL : &resource->limits[3];
108 ipa_resource_config_common(ipa, resource_type, reg,
109 &resource->limits[2], ylimits);
110 if (group_count < 5)
111 return;
112
113 reg = ipa_reg(ipa, SRC_RSRC_GRP_45_RSRC_TYPE);
114 ylimits = group_count == 5 ? NULL : &resource->limits[5];
115 ipa_resource_config_common(ipa, resource_type, reg,
116 &resource->limits[4], ylimits);
117 if (group_count < 7)
118 return;
119
120 reg = ipa_reg(ipa, SRC_RSRC_GRP_67_RSRC_TYPE);
121 ylimits = group_count == 7 ? NULL : &resource->limits[7];
122 ipa_resource_config_common(ipa, resource_type, reg,
123 &resource->limits[6], ylimits);
124}
125
126static void ipa_resource_config_dst(struct ipa *ipa, u32 resource_type,
127 const struct ipa_resource_data *data)
128{
129 u32 group_count = data->rsrc_group_dst_count;
130 const struct ipa_resource_limits *ylimits;
131 const struct ipa_resource *resource;
132 const struct reg *reg;
133
134 resource = &data->resource_dst[resource_type];
135
136 reg = ipa_reg(ipa, DST_RSRC_GRP_01_RSRC_TYPE);
137 ylimits = group_count == 1 ? NULL : &resource->limits[1];
138 ipa_resource_config_common(ipa, resource_type, reg,
139 &resource->limits[0], ylimits);
140 if (group_count < 3)
141 return;
142
143 reg = ipa_reg(ipa, DST_RSRC_GRP_23_RSRC_TYPE);
144 ylimits = group_count == 3 ? NULL : &resource->limits[3];
145 ipa_resource_config_common(ipa, resource_type, reg,
146 &resource->limits[2], ylimits);
147 if (group_count < 5)
148 return;
149
150 reg = ipa_reg(ipa, DST_RSRC_GRP_45_RSRC_TYPE);
151 ylimits = group_count == 5 ? NULL : &resource->limits[5];
152 ipa_resource_config_common(ipa, resource_type, reg,
153 &resource->limits[4], ylimits);
154 if (group_count < 7)
155 return;
156
157 reg = ipa_reg(ipa, DST_RSRC_GRP_67_RSRC_TYPE);
158 ylimits = group_count == 7 ? NULL : &resource->limits[7];
159 ipa_resource_config_common(ipa, resource_type, reg,
160 &resource->limits[6], ylimits);
161}
162
163/* Configure resources; there is no ipa_resource_deconfig() */
164int ipa_resource_config(struct ipa *ipa, const struct ipa_resource_data *data)
165{
166 u32 i;
167
168 if (!ipa_resource_limits_valid(ipa, data))
169 return -EINVAL;
170
171 for (i = 0; i < data->resource_src_count; i++)
172 ipa_resource_config_src(ipa, i, data);
173
174 for (i = 0; i < data->resource_dst_count; i++)
175 ipa_resource_config_dst(ipa, i, data);
176
177 return 0;
178}
1// SPDX-License-Identifier: GPL-2.0
2
3/* Copyright (c) 2012-2018, The Linux Foundation. All rights reserved.
4 * Copyright (C) 2018-2024 Linaro Ltd.
5 */
6
7#include <linux/types.h>
8
9#include "ipa.h"
10#include "ipa_data.h"
11#include "ipa_reg.h"
12#include "ipa_resource.h"
13
14/**
15 * DOC: IPA Resources
16 *
17 * The IPA manages a set of resources internally for various purposes.
18 * A given IPA version has a fixed number of resource types, and a fixed
19 * total number of resources of each type. "Source" resource types
20 * are separate from "destination" resource types.
21 *
22 * Each version of IPA also has some number of resource groups. Each
23 * endpoint is assigned to a resource group, and all endpoints in the
24 * same group share pools of each type of resource. A subset of the
25 * total resources of each type is assigned for use by each group.
26 */
27
28static bool ipa_resource_limits_valid(struct ipa *ipa,
29 const struct ipa_resource_data *data)
30{
31 u32 group_count;
32 u32 i;
33 u32 j;
34
35 /* We program at most 8 source or destination resource group limits */
36 BUILD_BUG_ON(IPA_RESOURCE_GROUP_MAX > 8);
37
38 group_count = data->rsrc_group_src_count;
39 if (!group_count || group_count > IPA_RESOURCE_GROUP_MAX)
40 return false;
41
42 /* Return an error if a non-zero resource limit is specified
43 * for a resource group not supported by hardware.
44 */
45 for (i = 0; i < data->resource_src_count; i++) {
46 const struct ipa_resource *resource;
47
48 resource = &data->resource_src[i];
49 for (j = group_count; j < IPA_RESOURCE_GROUP_MAX; j++)
50 if (resource->limits[j].min || resource->limits[j].max)
51 return false;
52 }
53
54 group_count = data->rsrc_group_dst_count;
55 if (!group_count || group_count > IPA_RESOURCE_GROUP_MAX)
56 return false;
57
58 for (i = 0; i < data->resource_dst_count; i++) {
59 const struct ipa_resource *resource;
60
61 resource = &data->resource_dst[i];
62 for (j = group_count; j < IPA_RESOURCE_GROUP_MAX; j++)
63 if (resource->limits[j].min || resource->limits[j].max)
64 return false;
65 }
66
67 return true;
68}
69
70static void
71ipa_resource_config_common(struct ipa *ipa, u32 resource_type,
72 const struct reg *reg,
73 const struct ipa_resource_limits *xlimits,
74 const struct ipa_resource_limits *ylimits)
75{
76 u32 val;
77
78 val = reg_encode(reg, X_MIN_LIM, xlimits->min);
79 val |= reg_encode(reg, X_MAX_LIM, xlimits->max);
80 if (ylimits) {
81 val |= reg_encode(reg, Y_MIN_LIM, ylimits->min);
82 val |= reg_encode(reg, Y_MAX_LIM, ylimits->max);
83 }
84
85 iowrite32(val, ipa->reg_virt + reg_n_offset(reg, resource_type));
86}
87
88static void ipa_resource_config_src(struct ipa *ipa, u32 resource_type,
89 const struct ipa_resource_data *data)
90{
91 u32 group_count = data->rsrc_group_src_count;
92 const struct ipa_resource_limits *ylimits;
93 const struct ipa_resource *resource;
94 const struct reg *reg;
95
96 resource = &data->resource_src[resource_type];
97
98 reg = ipa_reg(ipa, SRC_RSRC_GRP_01_RSRC_TYPE);
99 ylimits = group_count == 1 ? NULL : &resource->limits[1];
100 ipa_resource_config_common(ipa, resource_type, reg,
101 &resource->limits[0], ylimits);
102 if (group_count < 3)
103 return;
104
105 reg = ipa_reg(ipa, SRC_RSRC_GRP_23_RSRC_TYPE);
106 ylimits = group_count == 3 ? NULL : &resource->limits[3];
107 ipa_resource_config_common(ipa, resource_type, reg,
108 &resource->limits[2], ylimits);
109 if (group_count < 5)
110 return;
111
112 reg = ipa_reg(ipa, SRC_RSRC_GRP_45_RSRC_TYPE);
113 ylimits = group_count == 5 ? NULL : &resource->limits[5];
114 ipa_resource_config_common(ipa, resource_type, reg,
115 &resource->limits[4], ylimits);
116 if (group_count < 7)
117 return;
118
119 reg = ipa_reg(ipa, SRC_RSRC_GRP_67_RSRC_TYPE);
120 ylimits = group_count == 7 ? NULL : &resource->limits[7];
121 ipa_resource_config_common(ipa, resource_type, reg,
122 &resource->limits[6], ylimits);
123}
124
125static void ipa_resource_config_dst(struct ipa *ipa, u32 resource_type,
126 const struct ipa_resource_data *data)
127{
128 u32 group_count = data->rsrc_group_dst_count;
129 const struct ipa_resource_limits *ylimits;
130 const struct ipa_resource *resource;
131 const struct reg *reg;
132
133 resource = &data->resource_dst[resource_type];
134
135 reg = ipa_reg(ipa, DST_RSRC_GRP_01_RSRC_TYPE);
136 ylimits = group_count == 1 ? NULL : &resource->limits[1];
137 ipa_resource_config_common(ipa, resource_type, reg,
138 &resource->limits[0], ylimits);
139 if (group_count < 3)
140 return;
141
142 reg = ipa_reg(ipa, DST_RSRC_GRP_23_RSRC_TYPE);
143 ylimits = group_count == 3 ? NULL : &resource->limits[3];
144 ipa_resource_config_common(ipa, resource_type, reg,
145 &resource->limits[2], ylimits);
146 if (group_count < 5)
147 return;
148
149 reg = ipa_reg(ipa, DST_RSRC_GRP_45_RSRC_TYPE);
150 ylimits = group_count == 5 ? NULL : &resource->limits[5];
151 ipa_resource_config_common(ipa, resource_type, reg,
152 &resource->limits[4], ylimits);
153 if (group_count < 7)
154 return;
155
156 reg = ipa_reg(ipa, DST_RSRC_GRP_67_RSRC_TYPE);
157 ylimits = group_count == 7 ? NULL : &resource->limits[7];
158 ipa_resource_config_common(ipa, resource_type, reg,
159 &resource->limits[6], ylimits);
160}
161
162/* Configure resources; there is no ipa_resource_deconfig() */
163int ipa_resource_config(struct ipa *ipa, const struct ipa_resource_data *data)
164{
165 u32 i;
166
167 if (!ipa_resource_limits_valid(ipa, data))
168 return -EINVAL;
169
170 for (i = 0; i < data->resource_src_count; i++)
171 ipa_resource_config_src(ipa, i, data);
172
173 for (i = 0; i < data->resource_dst_count; i++)
174 ipa_resource_config_dst(ipa, i, data);
175
176 return 0;
177}