xref: /openbmc/linux/drivers/net/ethernet/freescale/fman/fman_muram.c (revision 4f2c0a4acffbec01079c28f839422e64ddeff004)
1*8585bdadSSean Anderson // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0-or-later
25adae51aSIgal Liberman /*
35adae51aSIgal Liberman  * Copyright 2008 - 2015 Freescale Semiconductor Inc.
45adae51aSIgal Liberman  */
55adae51aSIgal Liberman 
65adae51aSIgal Liberman #include "fman_muram.h"
75adae51aSIgal Liberman 
85adae51aSIgal Liberman #include <linux/io.h>
95adae51aSIgal Liberman #include <linux/slab.h>
105adae51aSIgal Liberman #include <linux/genalloc.h>
115adae51aSIgal Liberman 
125adae51aSIgal Liberman struct muram_info {
135adae51aSIgal Liberman 	struct gen_pool *pool;
145adae51aSIgal Liberman 	void __iomem *vbase;
155adae51aSIgal Liberman 	size_t size;
165adae51aSIgal Liberman 	phys_addr_t pbase;
175adae51aSIgal Liberman };
185adae51aSIgal Liberman 
fman_muram_vbase_to_offset(struct muram_info * muram,unsigned long vaddr)195adae51aSIgal Liberman static unsigned long fman_muram_vbase_to_offset(struct muram_info *muram,
205adae51aSIgal Liberman 						unsigned long vaddr)
215adae51aSIgal Liberman {
225adae51aSIgal Liberman 	return vaddr - (unsigned long)muram->vbase;
235adae51aSIgal Liberman }
245adae51aSIgal Liberman 
255adae51aSIgal Liberman /**
265adae51aSIgal Liberman  * fman_muram_init
275adae51aSIgal Liberman  * @base:	Pointer to base of memory mapped FM-MURAM.
285adae51aSIgal Liberman  * @size:	Size of the FM-MURAM partition.
295adae51aSIgal Liberman  *
305adae51aSIgal Liberman  * Creates partition in the MURAM.
315adae51aSIgal Liberman  * The routine returns a pointer to the MURAM partition.
325adae51aSIgal Liberman  * This pointer must be passed as to all other FM-MURAM function calls.
335adae51aSIgal Liberman  * No actual initialization or configuration of FM_MURAM hardware is done by
345adae51aSIgal Liberman  * this routine.
355adae51aSIgal Liberman  *
365adae51aSIgal Liberman  * Return: pointer to FM-MURAM object, or NULL for Failure.
375adae51aSIgal Liberman  */
fman_muram_init(phys_addr_t base,size_t size)385adae51aSIgal Liberman struct muram_info *fman_muram_init(phys_addr_t base, size_t size)
395adae51aSIgal Liberman {
405adae51aSIgal Liberman 	struct muram_info *muram;
415adae51aSIgal Liberman 	void __iomem *vaddr;
425adae51aSIgal Liberman 	int ret;
435adae51aSIgal Liberman 
445adae51aSIgal Liberman 	muram = kzalloc(sizeof(*muram), GFP_KERNEL);
455adae51aSIgal Liberman 	if (!muram)
465adae51aSIgal Liberman 		return NULL;
475adae51aSIgal Liberman 
485adae51aSIgal Liberman 	muram->pool = gen_pool_create(ilog2(64), -1);
495adae51aSIgal Liberman 	if (!muram->pool) {
505adae51aSIgal Liberman 		pr_err("%s(): MURAM pool create failed\n", __func__);
515adae51aSIgal Liberman 		goto  muram_free;
525adae51aSIgal Liberman 	}
535adae51aSIgal Liberman 
545adae51aSIgal Liberman 	vaddr = ioremap(base, size);
555adae51aSIgal Liberman 	if (!vaddr) {
565adae51aSIgal Liberman 		pr_err("%s(): MURAM ioremap failed\n", __func__);
575adae51aSIgal Liberman 		goto pool_destroy;
585adae51aSIgal Liberman 	}
595adae51aSIgal Liberman 
605adae51aSIgal Liberman 	ret = gen_pool_add_virt(muram->pool, (unsigned long)vaddr,
615adae51aSIgal Liberman 				base, size, -1);
625adae51aSIgal Liberman 	if (ret < 0) {
635adae51aSIgal Liberman 		pr_err("%s(): MURAM pool add failed\n", __func__);
645adae51aSIgal Liberman 		iounmap(vaddr);
655adae51aSIgal Liberman 		goto pool_destroy;
665adae51aSIgal Liberman 	}
675adae51aSIgal Liberman 
685adae51aSIgal Liberman 	memset_io(vaddr, 0, (int)size);
695adae51aSIgal Liberman 
705adae51aSIgal Liberman 	muram->vbase = vaddr;
715adae51aSIgal Liberman 	muram->pbase = base;
725adae51aSIgal Liberman 	return muram;
735adae51aSIgal Liberman 
745adae51aSIgal Liberman pool_destroy:
755adae51aSIgal Liberman 	gen_pool_destroy(muram->pool);
765adae51aSIgal Liberman muram_free:
775adae51aSIgal Liberman 	kfree(muram);
785adae51aSIgal Liberman 	return NULL;
795adae51aSIgal Liberman }
805adae51aSIgal Liberman 
815adae51aSIgal Liberman /**
825adae51aSIgal Liberman  * fman_muram_offset_to_vbase
835adae51aSIgal Liberman  * @muram:	FM-MURAM module pointer.
845adae51aSIgal Liberman  * @offset:	the offset of the memory block
855adae51aSIgal Liberman  *
865adae51aSIgal Liberman  * Gives the address of the memory region from specific offset
875adae51aSIgal Liberman  *
885adae51aSIgal Liberman  * Return: The address of the memory block
895adae51aSIgal Liberman  */
fman_muram_offset_to_vbase(struct muram_info * muram,unsigned long offset)905adae51aSIgal Liberman unsigned long fman_muram_offset_to_vbase(struct muram_info *muram,
915adae51aSIgal Liberman 					 unsigned long offset)
925adae51aSIgal Liberman {
935adae51aSIgal Liberman 	return offset + (unsigned long)muram->vbase;
945adae51aSIgal Liberman }
955adae51aSIgal Liberman 
965adae51aSIgal Liberman /**
975adae51aSIgal Liberman  * fman_muram_alloc
985adae51aSIgal Liberman  * @muram:	FM-MURAM module pointer.
995adae51aSIgal Liberman  * @size:	Size of the memory to be allocated.
1005adae51aSIgal Liberman  *
1015adae51aSIgal Liberman  * Allocate some memory from FM-MURAM partition.
1025adae51aSIgal Liberman  *
1035adae51aSIgal Liberman  * Return: address of the allocated memory; NULL otherwise.
1045adae51aSIgal Liberman  */
fman_muram_alloc(struct muram_info * muram,size_t size)105287980e4SArnd Bergmann unsigned long fman_muram_alloc(struct muram_info *muram, size_t size)
1065adae51aSIgal Liberman {
1075adae51aSIgal Liberman 	unsigned long vaddr;
1085adae51aSIgal Liberman 
1095adae51aSIgal Liberman 	vaddr = gen_pool_alloc(muram->pool, size);
1105adae51aSIgal Liberman 	if (!vaddr)
1115adae51aSIgal Liberman 		return -ENOMEM;
1125adae51aSIgal Liberman 
1135adae51aSIgal Liberman 	memset_io((void __iomem *)vaddr, 0, size);
1145adae51aSIgal Liberman 
1155adae51aSIgal Liberman 	return fman_muram_vbase_to_offset(muram, vaddr);
1165adae51aSIgal Liberman }
1175adae51aSIgal Liberman 
1185adae51aSIgal Liberman /**
1195adae51aSIgal Liberman  * fman_muram_free_mem
120d0ea5cbdSJesse Brandeburg  * @muram:	FM-MURAM module pointer.
121d0ea5cbdSJesse Brandeburg  * @offset:	offset of the memory region to be freed.
122d0ea5cbdSJesse Brandeburg  * @size:	size of the memory to be freed.
1235adae51aSIgal Liberman  *
1245adae51aSIgal Liberman  * Free an allocated memory from FM-MURAM partition.
1255adae51aSIgal Liberman  */
fman_muram_free_mem(struct muram_info * muram,unsigned long offset,size_t size)1268536aa06SMadalin Bucur void fman_muram_free_mem(struct muram_info *muram, unsigned long offset,
1278536aa06SMadalin Bucur 			 size_t size)
1285adae51aSIgal Liberman {
1295adae51aSIgal Liberman 	unsigned long addr = fman_muram_offset_to_vbase(muram, offset);
1305adae51aSIgal Liberman 
1315adae51aSIgal Liberman 	gen_pool_free(muram->pool, addr, size);
1325adae51aSIgal Liberman }
133