xref: /OK3568_Linux_fs/kernel/drivers/pci/endpoint/pci-epc-mem.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * PCI Endpoint *Controller* Address Space Management
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  * Copyright (C) 2017 Texas Instruments
6*4882a593Smuzhiyun  * Author: Kishon Vijay Abraham I <kishon@ti.com>
7*4882a593Smuzhiyun  */
8*4882a593Smuzhiyun 
9*4882a593Smuzhiyun #include <linux/io.h>
10*4882a593Smuzhiyun #include <linux/module.h>
11*4882a593Smuzhiyun #include <linux/slab.h>
12*4882a593Smuzhiyun 
13*4882a593Smuzhiyun #include <linux/pci-epc.h>
14*4882a593Smuzhiyun 
15*4882a593Smuzhiyun /**
16*4882a593Smuzhiyun  * pci_epc_mem_get_order() - determine the allocation order of a memory size
17*4882a593Smuzhiyun  * @mem: address space of the endpoint controller
18*4882a593Smuzhiyun  * @size: the size for which to get the order
19*4882a593Smuzhiyun  *
20*4882a593Smuzhiyun  * Reimplement get_order() for mem->page_size since the generic get_order
21*4882a593Smuzhiyun  * always gets order with a constant PAGE_SIZE.
22*4882a593Smuzhiyun  */
pci_epc_mem_get_order(struct pci_epc_mem * mem,size_t size)23*4882a593Smuzhiyun static int pci_epc_mem_get_order(struct pci_epc_mem *mem, size_t size)
24*4882a593Smuzhiyun {
25*4882a593Smuzhiyun 	int order;
26*4882a593Smuzhiyun 	unsigned int page_shift = ilog2(mem->window.page_size);
27*4882a593Smuzhiyun 
28*4882a593Smuzhiyun 	size--;
29*4882a593Smuzhiyun 	size >>= page_shift;
30*4882a593Smuzhiyun #if BITS_PER_LONG == 32
31*4882a593Smuzhiyun 	order = fls(size);
32*4882a593Smuzhiyun #else
33*4882a593Smuzhiyun 	order = fls64(size);
34*4882a593Smuzhiyun #endif
35*4882a593Smuzhiyun 	return order;
36*4882a593Smuzhiyun }
37*4882a593Smuzhiyun 
38*4882a593Smuzhiyun /**
39*4882a593Smuzhiyun  * pci_epc_multi_mem_init() - initialize the pci_epc_mem structure
40*4882a593Smuzhiyun  * @epc: the EPC device that invoked pci_epc_mem_init
41*4882a593Smuzhiyun  * @windows: pointer to windows supported by the device
42*4882a593Smuzhiyun  * @num_windows: number of windows device supports
43*4882a593Smuzhiyun  *
44*4882a593Smuzhiyun  * Invoke to initialize the pci_epc_mem structure used by the
45*4882a593Smuzhiyun  * endpoint functions to allocate mapped PCI address.
46*4882a593Smuzhiyun  */
pci_epc_multi_mem_init(struct pci_epc * epc,struct pci_epc_mem_window * windows,unsigned int num_windows)47*4882a593Smuzhiyun int pci_epc_multi_mem_init(struct pci_epc *epc,
48*4882a593Smuzhiyun 			   struct pci_epc_mem_window *windows,
49*4882a593Smuzhiyun 			   unsigned int num_windows)
50*4882a593Smuzhiyun {
51*4882a593Smuzhiyun 	struct pci_epc_mem *mem = NULL;
52*4882a593Smuzhiyun 	unsigned long *bitmap = NULL;
53*4882a593Smuzhiyun 	unsigned int page_shift;
54*4882a593Smuzhiyun 	size_t page_size;
55*4882a593Smuzhiyun 	int bitmap_size;
56*4882a593Smuzhiyun 	int pages;
57*4882a593Smuzhiyun 	int ret;
58*4882a593Smuzhiyun 	int i;
59*4882a593Smuzhiyun 
60*4882a593Smuzhiyun 	epc->num_windows = 0;
61*4882a593Smuzhiyun 
62*4882a593Smuzhiyun 	if (!windows || !num_windows)
63*4882a593Smuzhiyun 		return -EINVAL;
64*4882a593Smuzhiyun 
65*4882a593Smuzhiyun 	epc->windows = kcalloc(num_windows, sizeof(*epc->windows), GFP_KERNEL);
66*4882a593Smuzhiyun 	if (!epc->windows)
67*4882a593Smuzhiyun 		return -ENOMEM;
68*4882a593Smuzhiyun 
69*4882a593Smuzhiyun 	for (i = 0; i < num_windows; i++) {
70*4882a593Smuzhiyun 		page_size = windows[i].page_size;
71*4882a593Smuzhiyun 		if (page_size < PAGE_SIZE)
72*4882a593Smuzhiyun 			page_size = PAGE_SIZE;
73*4882a593Smuzhiyun 		page_shift = ilog2(page_size);
74*4882a593Smuzhiyun 		pages = windows[i].size >> page_shift;
75*4882a593Smuzhiyun 		bitmap_size = BITS_TO_LONGS(pages) * sizeof(long);
76*4882a593Smuzhiyun 
77*4882a593Smuzhiyun 		mem = kzalloc(sizeof(*mem), GFP_KERNEL);
78*4882a593Smuzhiyun 		if (!mem) {
79*4882a593Smuzhiyun 			ret = -ENOMEM;
80*4882a593Smuzhiyun 			i--;
81*4882a593Smuzhiyun 			goto err_mem;
82*4882a593Smuzhiyun 		}
83*4882a593Smuzhiyun 
84*4882a593Smuzhiyun 		bitmap = kzalloc(bitmap_size, GFP_KERNEL);
85*4882a593Smuzhiyun 		if (!bitmap) {
86*4882a593Smuzhiyun 			ret = -ENOMEM;
87*4882a593Smuzhiyun 			kfree(mem);
88*4882a593Smuzhiyun 			i--;
89*4882a593Smuzhiyun 			goto err_mem;
90*4882a593Smuzhiyun 		}
91*4882a593Smuzhiyun 
92*4882a593Smuzhiyun 		mem->window.phys_base = windows[i].phys_base;
93*4882a593Smuzhiyun 		mem->window.size = windows[i].size;
94*4882a593Smuzhiyun 		mem->window.page_size = page_size;
95*4882a593Smuzhiyun 		mem->bitmap = bitmap;
96*4882a593Smuzhiyun 		mem->pages = pages;
97*4882a593Smuzhiyun 		mutex_init(&mem->lock);
98*4882a593Smuzhiyun 		epc->windows[i] = mem;
99*4882a593Smuzhiyun 	}
100*4882a593Smuzhiyun 
101*4882a593Smuzhiyun 	epc->mem = epc->windows[0];
102*4882a593Smuzhiyun 	epc->num_windows = num_windows;
103*4882a593Smuzhiyun 
104*4882a593Smuzhiyun 	return 0;
105*4882a593Smuzhiyun 
106*4882a593Smuzhiyun err_mem:
107*4882a593Smuzhiyun 	for (; i >= 0; i--) {
108*4882a593Smuzhiyun 		mem = epc->windows[i];
109*4882a593Smuzhiyun 		kfree(mem->bitmap);
110*4882a593Smuzhiyun 		kfree(mem);
111*4882a593Smuzhiyun 	}
112*4882a593Smuzhiyun 	kfree(epc->windows);
113*4882a593Smuzhiyun 
114*4882a593Smuzhiyun 	return ret;
115*4882a593Smuzhiyun }
116*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(pci_epc_multi_mem_init);
117*4882a593Smuzhiyun 
pci_epc_mem_init(struct pci_epc * epc,phys_addr_t base,size_t size,size_t page_size)118*4882a593Smuzhiyun int pci_epc_mem_init(struct pci_epc *epc, phys_addr_t base,
119*4882a593Smuzhiyun 		     size_t size, size_t page_size)
120*4882a593Smuzhiyun {
121*4882a593Smuzhiyun 	struct pci_epc_mem_window mem_window;
122*4882a593Smuzhiyun 
123*4882a593Smuzhiyun 	mem_window.phys_base = base;
124*4882a593Smuzhiyun 	mem_window.size = size;
125*4882a593Smuzhiyun 	mem_window.page_size = page_size;
126*4882a593Smuzhiyun 
127*4882a593Smuzhiyun 	return pci_epc_multi_mem_init(epc, &mem_window, 1);
128*4882a593Smuzhiyun }
129*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(pci_epc_mem_init);
130*4882a593Smuzhiyun 
131*4882a593Smuzhiyun /**
132*4882a593Smuzhiyun  * pci_epc_mem_exit() - cleanup the pci_epc_mem structure
133*4882a593Smuzhiyun  * @epc: the EPC device that invoked pci_epc_mem_exit
134*4882a593Smuzhiyun  *
135*4882a593Smuzhiyun  * Invoke to cleanup the pci_epc_mem structure allocated in
136*4882a593Smuzhiyun  * pci_epc_mem_init().
137*4882a593Smuzhiyun  */
pci_epc_mem_exit(struct pci_epc * epc)138*4882a593Smuzhiyun void pci_epc_mem_exit(struct pci_epc *epc)
139*4882a593Smuzhiyun {
140*4882a593Smuzhiyun 	struct pci_epc_mem *mem;
141*4882a593Smuzhiyun 	int i;
142*4882a593Smuzhiyun 
143*4882a593Smuzhiyun 	if (!epc->num_windows)
144*4882a593Smuzhiyun 		return;
145*4882a593Smuzhiyun 
146*4882a593Smuzhiyun 	for (i = 0; i < epc->num_windows; i++) {
147*4882a593Smuzhiyun 		mem = epc->windows[i];
148*4882a593Smuzhiyun 		kfree(mem->bitmap);
149*4882a593Smuzhiyun 		kfree(mem);
150*4882a593Smuzhiyun 	}
151*4882a593Smuzhiyun 	kfree(epc->windows);
152*4882a593Smuzhiyun 
153*4882a593Smuzhiyun 	epc->windows = NULL;
154*4882a593Smuzhiyun 	epc->mem = NULL;
155*4882a593Smuzhiyun 	epc->num_windows = 0;
156*4882a593Smuzhiyun }
157*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(pci_epc_mem_exit);
158*4882a593Smuzhiyun 
159*4882a593Smuzhiyun /**
160*4882a593Smuzhiyun  * pci_epc_mem_alloc_addr() - allocate memory address from EPC addr space
161*4882a593Smuzhiyun  * @epc: the EPC device on which memory has to be allocated
162*4882a593Smuzhiyun  * @phys_addr: populate the allocated physical address here
163*4882a593Smuzhiyun  * @size: the size of the address space that has to be allocated
164*4882a593Smuzhiyun  *
165*4882a593Smuzhiyun  * Invoke to allocate memory address from the EPC address space. This
166*4882a593Smuzhiyun  * is usually done to map the remote RC address into the local system.
167*4882a593Smuzhiyun  */
pci_epc_mem_alloc_addr(struct pci_epc * epc,phys_addr_t * phys_addr,size_t size)168*4882a593Smuzhiyun void __iomem *pci_epc_mem_alloc_addr(struct pci_epc *epc,
169*4882a593Smuzhiyun 				     phys_addr_t *phys_addr, size_t size)
170*4882a593Smuzhiyun {
171*4882a593Smuzhiyun 	void __iomem *virt_addr = NULL;
172*4882a593Smuzhiyun 	struct pci_epc_mem *mem;
173*4882a593Smuzhiyun 	unsigned int page_shift;
174*4882a593Smuzhiyun 	size_t align_size;
175*4882a593Smuzhiyun 	int pageno;
176*4882a593Smuzhiyun 	int order;
177*4882a593Smuzhiyun 	int i;
178*4882a593Smuzhiyun 
179*4882a593Smuzhiyun 	for (i = 0; i < epc->num_windows; i++) {
180*4882a593Smuzhiyun 		mem = epc->windows[i];
181*4882a593Smuzhiyun 		mutex_lock(&mem->lock);
182*4882a593Smuzhiyun 		align_size = ALIGN(size, mem->window.page_size);
183*4882a593Smuzhiyun 		order = pci_epc_mem_get_order(mem, align_size);
184*4882a593Smuzhiyun 
185*4882a593Smuzhiyun 		pageno = bitmap_find_free_region(mem->bitmap, mem->pages,
186*4882a593Smuzhiyun 						 order);
187*4882a593Smuzhiyun 		if (pageno >= 0) {
188*4882a593Smuzhiyun 			page_shift = ilog2(mem->window.page_size);
189*4882a593Smuzhiyun 			*phys_addr = mem->window.phys_base +
190*4882a593Smuzhiyun 				((phys_addr_t)pageno << page_shift);
191*4882a593Smuzhiyun 			virt_addr = ioremap(*phys_addr, align_size);
192*4882a593Smuzhiyun 			if (!virt_addr) {
193*4882a593Smuzhiyun 				bitmap_release_region(mem->bitmap,
194*4882a593Smuzhiyun 						      pageno, order);
195*4882a593Smuzhiyun 				mutex_unlock(&mem->lock);
196*4882a593Smuzhiyun 				continue;
197*4882a593Smuzhiyun 			}
198*4882a593Smuzhiyun 			mutex_unlock(&mem->lock);
199*4882a593Smuzhiyun 			return virt_addr;
200*4882a593Smuzhiyun 		}
201*4882a593Smuzhiyun 		mutex_unlock(&mem->lock);
202*4882a593Smuzhiyun 	}
203*4882a593Smuzhiyun 
204*4882a593Smuzhiyun 	return virt_addr;
205*4882a593Smuzhiyun }
206*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(pci_epc_mem_alloc_addr);
207*4882a593Smuzhiyun 
pci_epc_get_matching_window(struct pci_epc * epc,phys_addr_t phys_addr)208*4882a593Smuzhiyun static struct pci_epc_mem *pci_epc_get_matching_window(struct pci_epc *epc,
209*4882a593Smuzhiyun 						       phys_addr_t phys_addr)
210*4882a593Smuzhiyun {
211*4882a593Smuzhiyun 	struct pci_epc_mem *mem;
212*4882a593Smuzhiyun 	int i;
213*4882a593Smuzhiyun 
214*4882a593Smuzhiyun 	for (i = 0; i < epc->num_windows; i++) {
215*4882a593Smuzhiyun 		mem = epc->windows[i];
216*4882a593Smuzhiyun 
217*4882a593Smuzhiyun 		if (phys_addr >= mem->window.phys_base &&
218*4882a593Smuzhiyun 		    phys_addr < (mem->window.phys_base + mem->window.size))
219*4882a593Smuzhiyun 			return mem;
220*4882a593Smuzhiyun 	}
221*4882a593Smuzhiyun 
222*4882a593Smuzhiyun 	return NULL;
223*4882a593Smuzhiyun }
224*4882a593Smuzhiyun 
225*4882a593Smuzhiyun /**
226*4882a593Smuzhiyun  * pci_epc_mem_free_addr() - free the allocated memory address
227*4882a593Smuzhiyun  * @epc: the EPC device on which memory was allocated
228*4882a593Smuzhiyun  * @phys_addr: the allocated physical address
229*4882a593Smuzhiyun  * @virt_addr: virtual address of the allocated mem space
230*4882a593Smuzhiyun  * @size: the size of the allocated address space
231*4882a593Smuzhiyun  *
232*4882a593Smuzhiyun  * Invoke to free the memory allocated using pci_epc_mem_alloc_addr.
233*4882a593Smuzhiyun  */
pci_epc_mem_free_addr(struct pci_epc * epc,phys_addr_t phys_addr,void __iomem * virt_addr,size_t size)234*4882a593Smuzhiyun void pci_epc_mem_free_addr(struct pci_epc *epc, phys_addr_t phys_addr,
235*4882a593Smuzhiyun 			   void __iomem *virt_addr, size_t size)
236*4882a593Smuzhiyun {
237*4882a593Smuzhiyun 	struct pci_epc_mem *mem;
238*4882a593Smuzhiyun 	unsigned int page_shift;
239*4882a593Smuzhiyun 	size_t page_size;
240*4882a593Smuzhiyun 	int pageno;
241*4882a593Smuzhiyun 	int order;
242*4882a593Smuzhiyun 
243*4882a593Smuzhiyun 	mem = pci_epc_get_matching_window(epc, phys_addr);
244*4882a593Smuzhiyun 	if (!mem) {
245*4882a593Smuzhiyun 		pr_err("failed to get matching window\n");
246*4882a593Smuzhiyun 		return;
247*4882a593Smuzhiyun 	}
248*4882a593Smuzhiyun 
249*4882a593Smuzhiyun 	page_size = mem->window.page_size;
250*4882a593Smuzhiyun 	page_shift = ilog2(page_size);
251*4882a593Smuzhiyun 	iounmap(virt_addr);
252*4882a593Smuzhiyun 	pageno = (phys_addr - mem->window.phys_base) >> page_shift;
253*4882a593Smuzhiyun 	size = ALIGN(size, page_size);
254*4882a593Smuzhiyun 	order = pci_epc_mem_get_order(mem, size);
255*4882a593Smuzhiyun 	mutex_lock(&mem->lock);
256*4882a593Smuzhiyun 	bitmap_release_region(mem->bitmap, pageno, order);
257*4882a593Smuzhiyun 	mutex_unlock(&mem->lock);
258*4882a593Smuzhiyun }
259*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(pci_epc_mem_free_addr);
260*4882a593Smuzhiyun 
261*4882a593Smuzhiyun MODULE_DESCRIPTION("PCI EPC Address Space Management");
262*4882a593Smuzhiyun MODULE_AUTHOR("Kishon Vijay Abraham I <kishon@ti.com>");
263*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
264