This source file includes following definitions.
- nffw_res_info_version_get
- nffw_res_flg_init_get
- nffw_fwinfo_loaded_get
- nffw_fwinfo_mip_cppid_get
- nffw_fwinfo_mip_mu_da_get
- nffw_fwinfo_mip_offset_get
- nffw_res_fwinfos
- nfp_nffw_info_open
- nfp_nffw_info_close
- nfp_nffw_info_fwid_first
- nfp_nffw_info_mip_first
1
2
3
4
5
6
7
8
9
10
11 #include <linux/kernel.h>
12 #include <linux/slab.h>
13
14 #include "nfp.h"
15 #include "nfp_cpp.h"
16 #include "nfp_nffw.h"
17 #include "nfp6000/nfp6000.h"
18
19
20
21
22 #define NFFW_FWID_EXT 3
23 #define NFFW_FWID_BASE 4
24
25 #define NFFW_FWID_ALL 255
26
27
28
29
30
31
32
33
34
35
36
37
38 #define NFFW_INFO_VERSION_CURRENT 2
39
40
41 #define NFFW_MEINFO_CNT_V1 120
42 #define NFFW_FWINFO_CNT_V1 120
43 #define NFFW_MEINFO_CNT_V2 200
44 #define NFFW_FWINFO_CNT_V2 200
45
46
47
48
49 struct nffw_meinfo {
50 __le32 ctxmask__fwid__meid;
51 };
52
53 struct nffw_fwinfo {
54 __le32 loaded__mu_da__mip_off_hi;
55 __le32 mip_cppid;
56 __le32 mip_offset_lo;
57 };
58
59 struct nfp_nffw_info_v1 {
60 struct nffw_meinfo meinfo[NFFW_MEINFO_CNT_V1];
61 struct nffw_fwinfo fwinfo[NFFW_FWINFO_CNT_V1];
62 };
63
64 struct nfp_nffw_info_v2 {
65 struct nffw_meinfo meinfo[NFFW_MEINFO_CNT_V2];
66 struct nffw_fwinfo fwinfo[NFFW_FWINFO_CNT_V2];
67 };
68
69
70 struct nfp_nffw_info_data {
71 __le32 flags[2];
72 union {
73 struct nfp_nffw_info_v1 v1;
74 struct nfp_nffw_info_v2 v2;
75 } info;
76 };
77
78 struct nfp_nffw_info {
79 struct nfp_cpp *cpp;
80 struct nfp_resource *res;
81
82 struct nfp_nffw_info_data fwinf;
83 };
84
85
86
87
88
89
90
91
92 static u32 nffw_res_info_version_get(const struct nfp_nffw_info_data *res)
93 {
94 return (le32_to_cpu(res->flags[0]) >> 16) & 0xfff;
95 }
96
97
98 static u32 nffw_res_flg_init_get(const struct nfp_nffw_info_data *res)
99 {
100 return (le32_to_cpu(res->flags[0]) >> 0) & 1;
101 }
102
103
104 static u32 nffw_fwinfo_loaded_get(const struct nffw_fwinfo *fi)
105 {
106 return (le32_to_cpu(fi->loaded__mu_da__mip_off_hi) >> 31) & 1;
107 }
108
109
110 static u32 nffw_fwinfo_mip_cppid_get(const struct nffw_fwinfo *fi)
111 {
112 return le32_to_cpu(fi->mip_cppid);
113 }
114
115
116 static u32 nffw_fwinfo_mip_mu_da_get(const struct nffw_fwinfo *fi)
117 {
118 return (le32_to_cpu(fi->loaded__mu_da__mip_off_hi) >> 8) & 1;
119 }
120
121
122 static u64 nffw_fwinfo_mip_offset_get(const struct nffw_fwinfo *fi)
123 {
124 u64 mip_off_hi = le32_to_cpu(fi->loaded__mu_da__mip_off_hi);
125
126 return (mip_off_hi & 0xFF) << 32 | le32_to_cpu(fi->mip_offset_lo);
127 }
128
129 static unsigned int
130 nffw_res_fwinfos(struct nfp_nffw_info_data *fwinf, struct nffw_fwinfo **arr)
131 {
132
133
134
135
136
137
138
139 switch (nffw_res_info_version_get(fwinf)) {
140 case 0:
141 case 1:
142 *arr = &fwinf->info.v1.fwinfo[0];
143 return NFFW_FWINFO_CNT_V1;
144 case 2:
145 *arr = &fwinf->info.v2.fwinfo[0];
146 return NFFW_FWINFO_CNT_V2;
147 default:
148 *arr = NULL;
149 return 0;
150 }
151 }
152
153
154
155
156
157
158
159 struct nfp_nffw_info *nfp_nffw_info_open(struct nfp_cpp *cpp)
160 {
161 struct nfp_nffw_info_data *fwinf;
162 struct nfp_nffw_info *state;
163 u32 info_ver;
164 int err;
165
166 state = kzalloc(sizeof(*state), GFP_KERNEL);
167 if (!state)
168 return ERR_PTR(-ENOMEM);
169
170 state->res = nfp_resource_acquire(cpp, NFP_RESOURCE_NFP_NFFW);
171 if (IS_ERR(state->res))
172 goto err_free;
173
174 fwinf = &state->fwinf;
175
176 if (sizeof(*fwinf) > nfp_resource_size(state->res))
177 goto err_release;
178
179 err = nfp_cpp_read(cpp, nfp_resource_cpp_id(state->res),
180 nfp_resource_address(state->res),
181 fwinf, sizeof(*fwinf));
182 if (err < (int)sizeof(*fwinf))
183 goto err_release;
184
185 if (!nffw_res_flg_init_get(fwinf))
186 goto err_release;
187
188 info_ver = nffw_res_info_version_get(fwinf);
189 if (info_ver > NFFW_INFO_VERSION_CURRENT)
190 goto err_release;
191
192 state->cpp = cpp;
193 return state;
194
195 err_release:
196 nfp_resource_release(state->res);
197 err_free:
198 kfree(state);
199 return ERR_PTR(-EIO);
200 }
201
202
203
204
205
206 void nfp_nffw_info_close(struct nfp_nffw_info *state)
207 {
208 nfp_resource_release(state->res);
209 kfree(state);
210 }
211
212
213
214
215
216
217
218 static struct nffw_fwinfo *nfp_nffw_info_fwid_first(struct nfp_nffw_info *state)
219 {
220 struct nffw_fwinfo *fwinfo;
221 unsigned int cnt, i;
222
223 cnt = nffw_res_fwinfos(&state->fwinf, &fwinfo);
224 if (!cnt)
225 return NULL;
226
227 for (i = 0; i < cnt; i++)
228 if (nffw_fwinfo_loaded_get(&fwinfo[i]))
229 return &fwinfo[i];
230
231 return NULL;
232 }
233
234
235
236
237
238
239
240
241
242 int nfp_nffw_info_mip_first(struct nfp_nffw_info *state, u32 *cpp_id, u64 *off)
243 {
244 struct nffw_fwinfo *fwinfo;
245
246 fwinfo = nfp_nffw_info_fwid_first(state);
247 if (!fwinfo)
248 return -EINVAL;
249
250 *cpp_id = nffw_fwinfo_mip_cppid_get(fwinfo);
251 *off = nffw_fwinfo_mip_offset_get(fwinfo);
252
253 if (nffw_fwinfo_mip_mu_da_get(fwinfo)) {
254 int locality_off = nfp_cpp_mu_locality_lsb(state->cpp);
255
256 *off &= ~(NFP_MU_ADDR_ACCESS_TYPE_MASK << locality_off);
257 *off |= NFP_MU_ADDR_ACCESS_TYPE_DIRECT << locality_off;
258 }
259
260 return 0;
261 }