1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Copyright (C) 2018-2022 Marvell International Ltd.
4  *
5  */
6 
7 #include <log.h>
8 #include <time.h>
9 #include <linux/delay.h>
10 
11 #include <mach/cvmx-regs.h>
12 #include <mach/cvmx-csr.h>
13 #include <mach/cvmx-bootmem.h>
14 #include <mach/octeon-model.h>
15 #include <mach/cvmx-fuse.h>
16 #include <mach/octeon-feature.h>
17 #include <mach/cvmx-qlm.h>
18 #include <mach/octeon_qlm.h>
19 #include <mach/cvmx-pcie.h>
20 #include <mach/cvmx-coremask.h>
21 #include <mach/cvmx-range.h>
22 
23 #include <mach/cvmx-global-resources.h>
24 #include <mach/cvmx-bootmem.h>
25 
26 #include <mach/cvmx-pki.h>
27 #include <mach/cvmx-helper.h>
28 #include <mach/cvmx-helper-board.h>
29 #include <mach/cvmx-helper-cfg.h>
30 
31 #define CVMX_MAX_GLOBAL_RESOURCES 128
32 #define CVMX_RESOURCES_ENTRIES_SIZE                                            \
33 	(sizeof(struct cvmx_global_resource_entry) * CVMX_MAX_GLOBAL_RESOURCES)
34 
35 /**
36  * This macro returns a member of the data
37  * structure. The argument "field" is the member name of the
38  * structure to read. The return type is a u64.
39  */
40 #define CVMX_GLOBAL_RESOURCES_GET_FIELD(field)                                 \
41 	__cvmx_struct_get_unsigned_field(                                      \
42 		__cvmx_global_resources_addr,                                  \
43 		offsetof(struct cvmx_global_resources, field),                 \
44 		SIZEOF_FIELD(struct cvmx_global_resources, field))
45 
46 /**
47  * This macro writes a member of the struct cvmx_global_resourcest
48  * structure. The argument "field" is the member name of the
49  * struct cvmx_global_resources to write.
50  */
51 #define CVMX_GLOBAL_RESOURCES_SET_FIELD(field, value)                          \
52 	__cvmx_struct_set_unsigned_field(                                      \
53 		__cvmx_global_resources_addr,                                  \
54 		offsetof(struct cvmx_global_resources, field),                 \
55 		SIZEOF_FIELD(struct cvmx_global_resources, field), value)
56 
57 /**
58  * This macro returns a member of the struct cvmx_global_resource_entry.
59  * The argument "field" is the member name of this structure.
60  * the return type is a u64. The "addr" parameter is the physical
61  * address of the structure.
62  */
63 #define CVMX_RESOURCE_ENTRY_GET_FIELD(addr, field)                             \
64 	__cvmx_struct_get_unsigned_field(                                      \
65 		addr, offsetof(struct cvmx_global_resource_entry, field),      \
66 		SIZEOF_FIELD(struct cvmx_global_resource_entry, field))
67 
68 /**
69  * This macro writes a member of the struct cvmx_global_resource_entry
70  * structure. The argument "field" is the member name of the
71  * struct cvmx_global_resource_entry to write. The "addr" parameter
72  * is the physical address of the structure.
73  */
74 #define CVMX_RESOURCE_ENTRY_SET_FIELD(addr, field, value)                      \
75 	__cvmx_struct_set_unsigned_field(                                      \
76 		addr, offsetof(struct cvmx_global_resource_entry, field),      \
77 		SIZEOF_FIELD(struct cvmx_global_resource_entry, field), value)
78 
79 #define CVMX_GET_RESOURCE_ENTRY(count)                                         \
80 	(__cvmx_global_resources_addr +                                        \
81 	 offsetof(struct cvmx_global_resources, resource_entry) +              \
82 	 (count * sizeof(struct cvmx_global_resource_entry)))
83 
84 #define CVMX_RESOURCE_TAG_SET_FIELD(addr, field, value)                        \
85 	__cvmx_struct_set_unsigned_field(                                      \
86 		addr, offsetof(struct global_resource_tag, field),             \
87 		SIZEOF_FIELD(struct global_resource_tag, field), value)
88 
89 #define CVMX_RESOURCE_TAG_GET_FIELD(addr, field)                               \
90 	__cvmx_struct_get_unsigned_field(                                      \
91 		addr, offsetof(struct global_resource_tag, field),             \
92 		SIZEOF_FIELD(struct global_resource_tag, field))
93 
94 #define MAX_RESOURCE_TAG_LEN		16
95 #define CVMX_GLOBAL_RESOURCE_NO_LOCKING (1)
96 
97 struct cvmx_global_resource_entry {
98 	struct global_resource_tag tag;
99 	u64 phys_addr;
100 	u64 size;
101 };
102 
103 struct cvmx_global_resources {
104 	u32 pad;
105 	u32 rlock;
106 	u64 entry_cnt;
107 	struct cvmx_global_resource_entry resource_entry[];
108 };
109 
110 /* Not the right place, putting it here for now */
111 u64 cvmx_app_id;
112 
113 /*
114  * Global named memory can be accessed anywhere even in 32-bit mode
115  */
116 static u64 __cvmx_global_resources_addr;
117 
118 /**
119  * This macro returns the size of a member of a structure.
120  */
121 #define SIZEOF_FIELD(s, field) sizeof(((s *)NULL)->field)
122 
123 /**
124  * This function is the implementation of the get macros defined
125  * for individual structure members. The argument are generated
126  * by the macros inorder to read only the needed memory.
127  *
128  * @param base   64bit physical address of the complete structure
129  * @param offset from the beginning of the structure to the member being
130  *               accessed.
131  * @param size   Size of the structure member.
132  *
133  * @return Value of the structure member promoted into a u64.
134  */
__cvmx_struct_get_unsigned_field(u64 base,int offset,int size)135 static inline u64 __cvmx_struct_get_unsigned_field(u64 base, int offset,
136 						   int size)
137 {
138 	base = (1ull << 63) | (base + offset);
139 	switch (size) {
140 	case 4:
141 		return cvmx_read64_uint32(base);
142 	case 8:
143 		return cvmx_read64_uint64(base);
144 	default:
145 		return 0;
146 	}
147 }
148 
149 /**
150  * This function is the implementation of the set macros defined
151  * for individual structure members. The argument are generated
152  * by the macros in order to write only the needed memory.
153  *
154  * @param base   64bit physical address of the complete structure
155  * @param offset from the beginning of the structure to the member being
156  *               accessed.
157  * @param size   Size of the structure member.
158  * @param value  Value to write into the structure
159  */
__cvmx_struct_set_unsigned_field(u64 base,int offset,int size,u64 value)160 static inline void __cvmx_struct_set_unsigned_field(u64 base, int offset,
161 						    int size, u64 value)
162 {
163 	base = (1ull << 63) | (base + offset);
164 	switch (size) {
165 	case 4:
166 		cvmx_write64_uint32(base, value);
167 		break;
168 	case 8:
169 		cvmx_write64_uint64(base, value);
170 		break;
171 	default:
172 		break;
173 	}
174 }
175 
176 /* Get the global resource lock. */
__cvmx_global_resource_lock(void)177 static inline void __cvmx_global_resource_lock(void)
178 {
179 	u64 lock_addr =
180 		(1ull << 63) | (__cvmx_global_resources_addr +
181 				offsetof(struct cvmx_global_resources, rlock));
182 	unsigned int tmp;
183 
184 	__asm__ __volatile__(".set noreorder\n"
185 			     "1: ll   %[tmp], 0(%[addr])\n"
186 			     "   bnez %[tmp], 1b\n"
187 			     "   li   %[tmp], 1\n"
188 			     "   sc   %[tmp], 0(%[addr])\n"
189 			     "   beqz %[tmp], 1b\n"
190 			     "   nop\n"
191 			     ".set reorder\n"
192 			     : [tmp] "=&r"(tmp)
193 			     : [addr] "r"(lock_addr)
194 			     : "memory");
195 }
196 
197 /* Release the global resource lock. */
__cvmx_global_resource_unlock(void)198 static inline void __cvmx_global_resource_unlock(void)
199 {
200 	u64 lock_addr =
201 		(1ull << 63) | (__cvmx_global_resources_addr +
202 				offsetof(struct cvmx_global_resources, rlock));
203 	CVMX_SYNCW;
204 	__asm__ __volatile__("sw $0, 0(%[addr])\n"
205 			     :
206 			     : [addr] "r"(lock_addr)
207 			     : "memory");
208 	CVMX_SYNCW;
209 }
210 
__cvmx_alloc_bootmem_for_global_resources(int sz)211 static u64 __cvmx_alloc_bootmem_for_global_resources(int sz)
212 {
213 	void *tmp;
214 
215 	tmp = cvmx_bootmem_alloc_range(sz, CVMX_CACHE_LINE_SIZE, 0, 0);
216 	return cvmx_ptr_to_phys(tmp);
217 }
218 
__cvmx_get_tagname(struct global_resource_tag * rtag,char * tagname)219 static inline void __cvmx_get_tagname(struct global_resource_tag *rtag,
220 				      char *tagname)
221 {
222 	int i, j, k;
223 
224 	j = 0;
225 	k = 8;
226 	for (i = 7; i >= 0; i--, j++, k++) {
227 		tagname[j] = (rtag->lo >> (i * 8)) & 0xff;
228 		tagname[k] = (rtag->hi >> (i * 8)) & 0xff;
229 	}
230 }
231 
__cvmx_global_resources_init(void)232 static u64 __cvmx_global_resources_init(void)
233 {
234 	struct cvmx_bootmem_named_block_desc *block_desc;
235 	int sz = sizeof(struct cvmx_global_resources) +
236 		 CVMX_RESOURCES_ENTRIES_SIZE;
237 	s64 tmp_phys;
238 	int count = 0;
239 	u64 base = 0;
240 
241 	cvmx_bootmem_lock();
242 
243 	block_desc = (struct cvmx_bootmem_named_block_desc *)
244 		__cvmx_bootmem_find_named_block_flags(
245 			CVMX_GLOBAL_RESOURCES_DATA_NAME,
246 			CVMX_BOOTMEM_FLAG_NO_LOCKING);
247 	if (!block_desc) {
248 		debug("%s: allocating global resources\n", __func__);
249 
250 		tmp_phys = cvmx_bootmem_phy_named_block_alloc(
251 			sz, 0, 0, CVMX_CACHE_LINE_SIZE,
252 			CVMX_GLOBAL_RESOURCES_DATA_NAME,
253 			CVMX_BOOTMEM_FLAG_NO_LOCKING);
254 		if (tmp_phys < 0) {
255 			cvmx_printf(
256 				"ERROR: %s: failed to allocate global resource name block. sz=%d\n",
257 				__func__, sz);
258 			goto end;
259 		}
260 		__cvmx_global_resources_addr = (u64)tmp_phys;
261 
262 		debug("%s: memset global resources %llu\n", __func__,
263 		      CAST_ULL(__cvmx_global_resources_addr));
264 
265 		base = (1ull << 63) | __cvmx_global_resources_addr;
266 		for (count = 0; count < (sz / 8); count++) {
267 			cvmx_write64_uint64(base, 0);
268 			base += 8;
269 		}
270 	} else {
271 		debug("%s:found global resource\n", __func__);
272 		__cvmx_global_resources_addr = block_desc->base_addr;
273 	}
274 end:
275 	cvmx_bootmem_unlock();
276 	debug("__cvmx_global_resources_addr=%llu sz=%d\n",
277 	      CAST_ULL(__cvmx_global_resources_addr), sz);
278 	return __cvmx_global_resources_addr;
279 }
280 
cvmx_get_global_resource(struct global_resource_tag tag,int no_lock)281 u64 cvmx_get_global_resource(struct global_resource_tag tag, int no_lock)
282 {
283 	u64 entry_cnt = 0;
284 	u64 resource_entry_addr = 0;
285 	int count = 0;
286 	u64 rphys_addr = 0;
287 	u64 tag_lo = 0, tag_hi = 0;
288 
289 	if (__cvmx_global_resources_addr == 0)
290 		__cvmx_global_resources_init();
291 	if (!no_lock)
292 		__cvmx_global_resource_lock();
293 
294 	entry_cnt = CVMX_GLOBAL_RESOURCES_GET_FIELD(entry_cnt);
295 	while (entry_cnt > 0) {
296 		resource_entry_addr = CVMX_GET_RESOURCE_ENTRY(count);
297 		tag_lo = CVMX_RESOURCE_TAG_GET_FIELD(resource_entry_addr, lo);
298 		tag_hi = CVMX_RESOURCE_TAG_GET_FIELD(resource_entry_addr, hi);
299 
300 		if (tag_lo == tag.lo && tag_hi == tag.hi) {
301 			debug("%s: Found global resource entry\n", __func__);
302 			break;
303 		}
304 		entry_cnt--;
305 		count++;
306 	}
307 
308 	if (entry_cnt == 0) {
309 		debug("%s: no matching global resource entry found\n",
310 		      __func__);
311 		if (!no_lock)
312 			__cvmx_global_resource_unlock();
313 		return 0;
314 	}
315 	rphys_addr =
316 		CVMX_RESOURCE_ENTRY_GET_FIELD(resource_entry_addr, phys_addr);
317 	if (!no_lock)
318 		__cvmx_global_resource_unlock();
319 
320 	return rphys_addr;
321 }
322 
cvmx_create_global_resource(struct global_resource_tag tag,u64 size,int no_lock,int * _new_)323 u64 cvmx_create_global_resource(struct global_resource_tag tag, u64 size,
324 				int no_lock, int *_new_)
325 {
326 	u64 entry_count = 0;
327 	u64 resource_entry_addr = 0;
328 	u64 phys_addr;
329 
330 	if (__cvmx_global_resources_addr == 0)
331 		__cvmx_global_resources_init();
332 
333 	if (!no_lock)
334 		__cvmx_global_resource_lock();
335 
336 	phys_addr =
337 		cvmx_get_global_resource(tag, CVMX_GLOBAL_RESOURCE_NO_LOCKING);
338 	if (phys_addr != 0) {
339 		/* we already have the resource, return it */
340 		*_new_ = 0;
341 		goto end;
342 	}
343 
344 	*_new_ = 1;
345 	entry_count = CVMX_GLOBAL_RESOURCES_GET_FIELD(entry_cnt);
346 	if (entry_count >= CVMX_MAX_GLOBAL_RESOURCES) {
347 		char tagname[MAX_RESOURCE_TAG_LEN + 1];
348 
349 		__cvmx_get_tagname(&tag, tagname);
350 		cvmx_printf(
351 			"ERROR: %s: reached global resources limit for %s\n",
352 			__func__, tagname);
353 		phys_addr = 0;
354 		goto end;
355 	}
356 
357 	/* Allocate bootmem for the resource*/
358 	phys_addr = __cvmx_alloc_bootmem_for_global_resources(size);
359 	if (!phys_addr) {
360 		char tagname[MAX_RESOURCE_TAG_LEN + 1];
361 
362 		__cvmx_get_tagname(&tag, tagname);
363 		debug("ERROR: %s: out of memory %s, size=%d\n", __func__,
364 		      tagname, (int)size);
365 		goto end;
366 	}
367 
368 	resource_entry_addr = CVMX_GET_RESOURCE_ENTRY(entry_count);
369 	CVMX_RESOURCE_ENTRY_SET_FIELD(resource_entry_addr, phys_addr,
370 				      phys_addr);
371 	CVMX_RESOURCE_ENTRY_SET_FIELD(resource_entry_addr, size, size);
372 	CVMX_RESOURCE_TAG_SET_FIELD(resource_entry_addr, lo, tag.lo);
373 	CVMX_RESOURCE_TAG_SET_FIELD(resource_entry_addr, hi, tag.hi);
374 	/* update entry_cnt */
375 	entry_count += 1;
376 	CVMX_GLOBAL_RESOURCES_SET_FIELD(entry_cnt, entry_count);
377 
378 end:
379 	if (!no_lock)
380 		__cvmx_global_resource_unlock();
381 
382 	return phys_addr;
383 }
384 
cvmx_create_global_resource_range(struct global_resource_tag tag,int nelements)385 int cvmx_create_global_resource_range(struct global_resource_tag tag,
386 				      int nelements)
387 {
388 	int sz = cvmx_range_memory_size(nelements);
389 	int _new_;
390 	u64 addr;
391 	int rv = 0;
392 
393 	if (__cvmx_global_resources_addr == 0)
394 		__cvmx_global_resources_init();
395 
396 	__cvmx_global_resource_lock();
397 	addr = cvmx_create_global_resource(tag, sz, 1, &_new_);
398 	if (!addr) {
399 		__cvmx_global_resource_unlock();
400 		return -1;
401 	}
402 	if (_new_)
403 		rv = cvmx_range_init(addr, nelements);
404 	__cvmx_global_resource_unlock();
405 	return rv;
406 }
407 
cvmx_allocate_global_resource_range(struct global_resource_tag tag,u64 owner,int nelements,int alignment)408 int cvmx_allocate_global_resource_range(struct global_resource_tag tag,
409 					u64 owner, int nelements, int alignment)
410 {
411 	u64 addr = cvmx_get_global_resource(tag, 1);
412 	int base;
413 
414 	if (addr == 0) {
415 		char tagname[256];
416 
417 		__cvmx_get_tagname(&tag, tagname);
418 		cvmx_printf("ERROR: %s: cannot find resource %s\n", __func__,
419 			    tagname);
420 		return -1;
421 	}
422 	__cvmx_global_resource_lock();
423 	base = cvmx_range_alloc(addr, owner, nelements, alignment);
424 	__cvmx_global_resource_unlock();
425 	return base;
426 }
427 
cvmx_resource_alloc_reverse(struct global_resource_tag tag,u64 owner)428 int cvmx_resource_alloc_reverse(struct global_resource_tag tag, u64 owner)
429 {
430 	u64 addr = cvmx_get_global_resource(tag, 1);
431 	int rv;
432 
433 	if (addr == 0) {
434 		char tagname[256];
435 
436 		__cvmx_get_tagname(&tag, tagname);
437 		debug("ERROR: cannot find resource %s\n", tagname);
438 		return -1;
439 	}
440 	__cvmx_global_resource_lock();
441 	rv = cvmx_range_alloc_ordered(addr, owner, 1, 1, 1);
442 	__cvmx_global_resource_unlock();
443 	return rv;
444 }
445 
cvmx_reserve_global_resource_range(struct global_resource_tag tag,u64 owner,int base,int nelements)446 int cvmx_reserve_global_resource_range(struct global_resource_tag tag,
447 				       u64 owner, int base, int nelements)
448 {
449 	u64 addr = cvmx_get_global_resource(tag, 1);
450 	int start;
451 
452 	__cvmx_global_resource_lock();
453 	start = cvmx_range_reserve(addr, owner, base, nelements);
454 	__cvmx_global_resource_unlock();
455 	return start;
456 }
457 
cvmx_free_global_resource_range_with_base(struct global_resource_tag tag,int base,int nelements)458 int cvmx_free_global_resource_range_with_base(struct global_resource_tag tag,
459 					      int base, int nelements)
460 {
461 	u64 addr = cvmx_get_global_resource(tag, 1);
462 	int rv;
463 
464 	/* Resource was not created, nothing to release */
465 	if (addr == 0)
466 		return 0;
467 
468 	__cvmx_global_resource_lock();
469 	rv = cvmx_range_free_with_base(addr, base, nelements);
470 	__cvmx_global_resource_unlock();
471 	return rv;
472 }
473 
cvmx_free_global_resource_range_multiple(struct global_resource_tag tag,int bases[],int nelements)474 int cvmx_free_global_resource_range_multiple(struct global_resource_tag tag,
475 					     int bases[], int nelements)
476 {
477 	u64 addr = cvmx_get_global_resource(tag, 1);
478 	int rv;
479 
480 	/* Resource was not created, nothing to release */
481 	if (addr == 0)
482 		return 0;
483 
484 	__cvmx_global_resource_lock();
485 	rv = cvmx_range_free_mutiple(addr, bases, nelements);
486 	__cvmx_global_resource_unlock();
487 	return rv;
488 }
489 
cvmx_app_id_init(void * bootmem)490 void cvmx_app_id_init(void *bootmem)
491 {
492 	u64 *p = (u64 *)bootmem;
493 
494 	*p = 0;
495 }
496 
cvmx_allocate_app_id(void)497 u64 cvmx_allocate_app_id(void)
498 {
499 	u64 *vptr;
500 
501 	vptr = (u64 *)cvmx_bootmem_alloc_named_range_once(sizeof(cvmx_app_id),
502 							  0, 1 << 31, 128,
503 							  "cvmx_app_id",
504 							  cvmx_app_id_init);
505 
506 	cvmx_app_id = __atomic_add_fetch(vptr, 1, __ATOMIC_SEQ_CST);
507 
508 	debug("CVMX_APP_ID = %lx\n", (unsigned long)cvmx_app_id);
509 	return cvmx_app_id;
510 }
511 
cvmx_get_app_id(void)512 u64 cvmx_get_app_id(void)
513 {
514 	if (cvmx_app_id == 0)
515 		cvmx_allocate_app_id();
516 	return cvmx_app_id;
517 }
518