blob: 2881e49a39571b47029d607cad4f429b5ce5a0f5 [file] [log] [blame]
Duy Truonge833aca2013-02-12 13:35:08 -08001/* Copyright (c) 2002,2007-2012, The Linux Foundation. All rights reserved.
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 *
12 */
Steve Mucklef132c6c2012-06-06 18:30:57 -070013
14#include <linux/export.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070015#include <linux/vmalloc.h>
16#include <linux/memory_alloc.h>
17#include <asm/cacheflush.h>
Anshuman Danieecd5202012-02-17 19:52:49 +053018#include <linux/slab.h>
19#include <linux/kmemleak.h>
Jordan Crouse89bd3232012-07-02 17:50:15 -060020#include <linux/highmem.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070021
22#include "kgsl.h"
23#include "kgsl_sharedmem.h"
24#include "kgsl_cffdump.h"
25#include "kgsl_device.h"
26
Jordan Crouse1b897cf2011-10-12 16:57:48 -060027/* An attribute for showing per-process memory statistics */
28struct kgsl_mem_entry_attribute {
29 struct attribute attr;
30 int memtype;
31 ssize_t (*show)(struct kgsl_process_private *priv,
32 int type, char *buf);
33};
34
35#define to_mem_entry_attr(a) \
36container_of(a, struct kgsl_mem_entry_attribute, attr)
37
38#define __MEM_ENTRY_ATTR(_type, _name, _show) \
39{ \
40 .attr = { .name = __stringify(_name), .mode = 0444 }, \
41 .memtype = _type, \
42 .show = _show, \
43}
44
45/*
46 * A structure to hold the attributes for a particular memory type.
47 * For each memory type in each process we store the current and maximum
48 * memory usage and display the counts in sysfs. This structure and
49 * the following macro allow us to simplify the definition for those
50 * adding new memory types
51 */
52
53struct mem_entry_stats {
54 int memtype;
55 struct kgsl_mem_entry_attribute attr;
56 struct kgsl_mem_entry_attribute max_attr;
57};
58
59
60#define MEM_ENTRY_STAT(_type, _name) \
61{ \
62 .memtype = _type, \
63 .attr = __MEM_ENTRY_ATTR(_type, _name, mem_entry_show), \
64 .max_attr = __MEM_ENTRY_ATTR(_type, _name##_max, \
65 mem_entry_max_show), \
66}
67
Jordan Crouse1b897cf2011-10-12 16:57:48 -060068/**
69 * Given a kobj, find the process structure attached to it
70 */
71
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070072static struct kgsl_process_private *
73_get_priv_from_kobj(struct kobject *kobj)
74{
75 struct kgsl_process_private *private;
76 unsigned long name;
77
78 if (!kobj)
79 return NULL;
80
81 if (sscanf(kobj->name, "%ld", &name) != 1)
82 return NULL;
83
84 list_for_each_entry(private, &kgsl_driver.process_list, list) {
85 if (private->pid == name)
86 return private;
87 }
88
89 return NULL;
90}
91
Jordan Crouse1b897cf2011-10-12 16:57:48 -060092/**
93 * Show the current amount of memory allocated for the given memtype
94 */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070095
96static ssize_t
Jordan Crouse1b897cf2011-10-12 16:57:48 -060097mem_entry_show(struct kgsl_process_private *priv, int type, char *buf)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070098{
Jordan Crouse1b897cf2011-10-12 16:57:48 -060099 return snprintf(buf, PAGE_SIZE, "%d\n", priv->stats[type].cur);
100}
101
102/**
103 * Show the maximum memory allocated for the given memtype through the life of
104 * the process
105 */
106
107static ssize_t
108mem_entry_max_show(struct kgsl_process_private *priv, int type, char *buf)
109{
110 return snprintf(buf, PAGE_SIZE, "%d\n", priv->stats[type].max);
111}
112
113
114static void mem_entry_sysfs_release(struct kobject *kobj)
115{
116}
117
118static ssize_t mem_entry_sysfs_show(struct kobject *kobj,
119 struct attribute *attr, char *buf)
120{
121 struct kgsl_mem_entry_attribute *pattr = to_mem_entry_attr(attr);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700122 struct kgsl_process_private *priv;
Jordan Crouse1b897cf2011-10-12 16:57:48 -0600123 ssize_t ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700124
125 mutex_lock(&kgsl_driver.process_mutex);
126 priv = _get_priv_from_kobj(kobj);
127
Jordan Crouse1b897cf2011-10-12 16:57:48 -0600128 if (priv && pattr->show)
129 ret = pattr->show(priv, pattr->memtype, buf);
130 else
131 ret = -EIO;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700132
133 mutex_unlock(&kgsl_driver.process_mutex);
Jordan Crouse1b897cf2011-10-12 16:57:48 -0600134 return ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700135}
136
Jordan Crouse1b897cf2011-10-12 16:57:48 -0600137static const struct sysfs_ops mem_entry_sysfs_ops = {
138 .show = mem_entry_sysfs_show,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700139};
140
Jordan Crouse1b897cf2011-10-12 16:57:48 -0600141static struct kobj_type ktype_mem_entry = {
142 .sysfs_ops = &mem_entry_sysfs_ops,
143 .default_attrs = NULL,
144 .release = mem_entry_sysfs_release
145};
146
147static struct mem_entry_stats mem_stats[] = {
148 MEM_ENTRY_STAT(KGSL_MEM_ENTRY_KERNEL, kernel),
149#ifdef CONFIG_ANDROID_PMEM
150 MEM_ENTRY_STAT(KGSL_MEM_ENTRY_PMEM, pmem),
151#endif
152#ifdef CONFIG_ASHMEM
153 MEM_ENTRY_STAT(KGSL_MEM_ENTRY_ASHMEM, ashmem),
154#endif
155 MEM_ENTRY_STAT(KGSL_MEM_ENTRY_USER, user),
Jordan Crouse8eab35a2011-10-12 16:57:48 -0600156#ifdef CONFIG_ION
Jeremy Gebbenff6eab02012-01-09 09:42:21 -0700157 MEM_ENTRY_STAT(KGSL_MEM_ENTRY_ION, ion),
Jordan Crouse8eab35a2011-10-12 16:57:48 -0600158#endif
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700159};
160
161void
162kgsl_process_uninit_sysfs(struct kgsl_process_private *private)
163{
Jordan Crouse1b897cf2011-10-12 16:57:48 -0600164 int i;
165
166 for (i = 0; i < ARRAY_SIZE(mem_stats); i++) {
167 sysfs_remove_file(&private->kobj, &mem_stats[i].attr.attr);
168 sysfs_remove_file(&private->kobj,
169 &mem_stats[i].max_attr.attr);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700170 }
Jordan Crouse1b897cf2011-10-12 16:57:48 -0600171
172 kobject_put(&private->kobj);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700173}
174
175void
176kgsl_process_init_sysfs(struct kgsl_process_private *private)
177{
178 unsigned char name[16];
Jordan Crouse1b897cf2011-10-12 16:57:48 -0600179 int i, ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700180
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700181 snprintf(name, sizeof(name), "%d", private->pid);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700182
Jordan Crouse1b897cf2011-10-12 16:57:48 -0600183 if (kobject_init_and_add(&private->kobj, &ktype_mem_entry,
184 kgsl_driver.prockobj, name))
185 return;
186
187 for (i = 0; i < ARRAY_SIZE(mem_stats); i++) {
188 /* We need to check the value of sysfs_create_file, but we
189 * don't really care if it passed or not */
190
191 ret = sysfs_create_file(&private->kobj,
192 &mem_stats[i].attr.attr);
193 ret = sysfs_create_file(&private->kobj,
194 &mem_stats[i].max_attr.attr);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700195 }
196}
197
198static int kgsl_drv_memstat_show(struct device *dev,
199 struct device_attribute *attr,
200 char *buf)
201{
202 unsigned int val = 0;
203
204 if (!strncmp(attr->attr.name, "vmalloc", 7))
205 val = kgsl_driver.stats.vmalloc;
206 else if (!strncmp(attr->attr.name, "vmalloc_max", 11))
207 val = kgsl_driver.stats.vmalloc_max;
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600208 else if (!strncmp(attr->attr.name, "page_alloc", 10))
209 val = kgsl_driver.stats.page_alloc;
210 else if (!strncmp(attr->attr.name, "page_alloc_max", 14))
211 val = kgsl_driver.stats.page_alloc_max;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700212 else if (!strncmp(attr->attr.name, "coherent", 8))
213 val = kgsl_driver.stats.coherent;
214 else if (!strncmp(attr->attr.name, "coherent_max", 12))
215 val = kgsl_driver.stats.coherent_max;
216 else if (!strncmp(attr->attr.name, "mapped", 6))
217 val = kgsl_driver.stats.mapped;
218 else if (!strncmp(attr->attr.name, "mapped_max", 10))
219 val = kgsl_driver.stats.mapped_max;
220
221 return snprintf(buf, PAGE_SIZE, "%u\n", val);
222}
223
224static int kgsl_drv_histogram_show(struct device *dev,
225 struct device_attribute *attr,
226 char *buf)
227{
228 int len = 0;
229 int i;
230
231 for (i = 0; i < 16; i++)
Jeremy Gebbena87bb862011-08-08 16:09:38 -0600232 len += snprintf(buf + len, PAGE_SIZE - len, "%d ",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700233 kgsl_driver.stats.histogram[i]);
234
Jeremy Gebbena87bb862011-08-08 16:09:38 -0600235 len += snprintf(buf + len, PAGE_SIZE - len, "\n");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700236 return len;
237}
238
239DEVICE_ATTR(vmalloc, 0444, kgsl_drv_memstat_show, NULL);
240DEVICE_ATTR(vmalloc_max, 0444, kgsl_drv_memstat_show, NULL);
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600241DEVICE_ATTR(page_alloc, 0444, kgsl_drv_memstat_show, NULL);
242DEVICE_ATTR(page_alloc_max, 0444, kgsl_drv_memstat_show, NULL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700243DEVICE_ATTR(coherent, 0444, kgsl_drv_memstat_show, NULL);
244DEVICE_ATTR(coherent_max, 0444, kgsl_drv_memstat_show, NULL);
245DEVICE_ATTR(mapped, 0444, kgsl_drv_memstat_show, NULL);
246DEVICE_ATTR(mapped_max, 0444, kgsl_drv_memstat_show, NULL);
247DEVICE_ATTR(histogram, 0444, kgsl_drv_histogram_show, NULL);
248
249static const struct device_attribute *drv_attr_list[] = {
250 &dev_attr_vmalloc,
251 &dev_attr_vmalloc_max,
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600252 &dev_attr_page_alloc,
253 &dev_attr_page_alloc_max,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700254 &dev_attr_coherent,
255 &dev_attr_coherent_max,
256 &dev_attr_mapped,
257 &dev_attr_mapped_max,
258 &dev_attr_histogram,
259 NULL
260};
261
262void
263kgsl_sharedmem_uninit_sysfs(void)
264{
265 kgsl_remove_device_sysfs_files(&kgsl_driver.virtdev, drv_attr_list);
266}
267
268int
269kgsl_sharedmem_init_sysfs(void)
270{
271 return kgsl_create_device_sysfs_files(&kgsl_driver.virtdev,
272 drv_attr_list);
273}
274
275#ifdef CONFIG_OUTER_CACHE
276static void _outer_cache_range_op(int op, unsigned long addr, size_t size)
277{
278 switch (op) {
279 case KGSL_CACHE_OP_FLUSH:
280 outer_flush_range(addr, addr + size);
281 break;
282 case KGSL_CACHE_OP_CLEAN:
283 outer_clean_range(addr, addr + size);
284 break;
285 case KGSL_CACHE_OP_INV:
286 outer_inv_range(addr, addr + size);
287 break;
288 }
289}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700290
Jordan Croused17e9aa2011-10-12 16:57:48 -0600291static void outer_cache_range_op_sg(struct scatterlist *sg, int sglen, int op)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700292{
Jordan Croused17e9aa2011-10-12 16:57:48 -0600293 struct scatterlist *s;
294 int i;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700295
Jordan Croused17e9aa2011-10-12 16:57:48 -0600296 for_each_sg(sg, s, sglen, i) {
Jeremy Gebben582fe312012-03-23 10:19:44 -0600297 unsigned int paddr = kgsl_get_sg_pa(s);
Jordan Croused17e9aa2011-10-12 16:57:48 -0600298 _outer_cache_range_op(op, paddr, s->length);
299 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700300}
301
Jordan Croused17e9aa2011-10-12 16:57:48 -0600302#else
303static void outer_cache_range_op_sg(struct scatterlist *sg, int sglen, int op)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700304{
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700305}
306#endif
307
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600308static int kgsl_page_alloc_vmfault(struct kgsl_memdesc *memdesc,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700309 struct vm_area_struct *vma,
310 struct vm_fault *vmf)
311{
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800312 int i, pgoff;
313 struct scatterlist *s = memdesc->sg;
314 unsigned int offset;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700315
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800316 offset = ((unsigned long) vmf->virtual_address - vma->vm_start);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700317
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800318 if (offset >= memdesc->size)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700319 return VM_FAULT_SIGBUS;
320
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800321 pgoff = offset >> PAGE_SHIFT;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700322
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800323 /*
324 * The sglist might be comprised of mixed blocks of memory depending
325 * on how many 64K pages were allocated. This means we have to do math
326 * to find the actual 4K page to map in user space
327 */
328
329 for (i = 0; i < memdesc->sglen; i++) {
330 int npages = s->length >> PAGE_SHIFT;
331
332 if (pgoff < npages) {
333 struct page *page = sg_page(s);
334
335 page = nth_page(page, pgoff);
336
337 get_page(page);
338 vmf->page = page;
339
340 return 0;
341 }
342
343 pgoff -= npages;
344 s = sg_next(s);
345 }
346
347 return VM_FAULT_SIGBUS;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700348}
349
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600350static int kgsl_page_alloc_vmflags(struct kgsl_memdesc *memdesc)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700351{
352 return VM_RESERVED | VM_DONTEXPAND;
353}
354
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600355static void kgsl_page_alloc_free(struct kgsl_memdesc *memdesc)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700356{
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600357 int i = 0;
358 struct scatterlist *sg;
Jordan Crouse7d3139b2012-05-18 10:05:02 -0600359 int sglen = memdesc->sglen;
360
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600361 kgsl_driver.stats.page_alloc -= memdesc->size;
Jordan Crouse7d3139b2012-05-18 10:05:02 -0600362
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600363 if (memdesc->hostptr) {
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600364 vunmap(memdesc->hostptr);
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600365 kgsl_driver.stats.vmalloc -= memdesc->size;
366 }
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600367 if (memdesc->sg)
Dhivya Subramanian24866ee2013-02-07 12:24:26 -0800368 for_each_sg(memdesc->sg, sg, sglen, i){
369 if (sg->length == 0)
370 break;
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800371 __free_pages(sg_page(sg), get_order(sg->length));
Dhivya Subramanian24866ee2013-02-07 12:24:26 -0800372 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700373}
374
375static int kgsl_contiguous_vmflags(struct kgsl_memdesc *memdesc)
376{
377 return VM_RESERVED | VM_IO | VM_PFNMAP | VM_DONTEXPAND;
378}
379
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600380/*
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600381 * kgsl_page_alloc_map_kernel - Map the memory in memdesc to kernel address
382 * space
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600383 *
384 * @memdesc - The memory descriptor which contains information about the memory
385 *
386 * Return: 0 on success else error code
387 */
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600388static int kgsl_page_alloc_map_kernel(struct kgsl_memdesc *memdesc)
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600389{
390 if (!memdesc->hostptr) {
391 pgprot_t page_prot = pgprot_writecombine(PAGE_KERNEL);
392 struct page **pages = NULL;
393 struct scatterlist *sg;
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800394 int npages = PAGE_ALIGN(memdesc->size) >> PAGE_SHIFT;
Jordan Crouse7d3139b2012-05-18 10:05:02 -0600395 int sglen = memdesc->sglen;
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800396 int i, count = 0;
Jordan Crouse7d3139b2012-05-18 10:05:02 -0600397
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600398 /* create a list of pages to call vmap */
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800399 pages = vmalloc(npages * sizeof(struct page *));
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600400 if (!pages) {
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800401 KGSL_CORE_ERR("vmalloc(%d) failed\n",
402 npages * sizeof(struct page *));
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600403 return -ENOMEM;
404 }
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800405
406 for_each_sg(memdesc->sg, sg, sglen, i) {
407 struct page *page = sg_page(sg);
408 int j;
409
410 for (j = 0; j < sg->length >> PAGE_SHIFT; j++)
411 pages[count++] = page++;
412 }
413
414
415 memdesc->hostptr = vmap(pages, count,
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600416 VM_IOREMAP, page_prot);
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600417 KGSL_STATS_ADD(memdesc->size, kgsl_driver.stats.vmalloc,
418 kgsl_driver.stats.vmalloc_max);
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800419 vfree(pages);
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600420 }
421 if (!memdesc->hostptr)
422 return -ENOMEM;
423
424 return 0;
425}
426
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700427static int kgsl_contiguous_vmfault(struct kgsl_memdesc *memdesc,
428 struct vm_area_struct *vma,
429 struct vm_fault *vmf)
430{
431 unsigned long offset, pfn;
432 int ret;
433
434 offset = ((unsigned long) vmf->virtual_address - vma->vm_start) >>
435 PAGE_SHIFT;
436
437 pfn = (memdesc->physaddr >> PAGE_SHIFT) + offset;
438 ret = vm_insert_pfn(vma, (unsigned long) vmf->virtual_address, pfn);
439
440 if (ret == -ENOMEM || ret == -EAGAIN)
441 return VM_FAULT_OOM;
442 else if (ret == -EFAULT)
443 return VM_FAULT_SIGBUS;
444
445 return VM_FAULT_NOPAGE;
446}
447
448static void kgsl_ebimem_free(struct kgsl_memdesc *memdesc)
449
450{
451 kgsl_driver.stats.coherent -= memdesc->size;
452 if (memdesc->hostptr)
453 iounmap(memdesc->hostptr);
454
455 free_contiguous_memory_by_paddr(memdesc->physaddr);
456}
457
Shubhraprakash Das387b6692012-08-11 18:26:29 -0700458static int kgsl_ebimem_map_kernel(struct kgsl_memdesc *memdesc)
459{
460 if (!memdesc->hostptr) {
461 memdesc->hostptr = ioremap(memdesc->physaddr, memdesc->size);
462 if (!memdesc->hostptr) {
463 KGSL_CORE_ERR("ioremap failed, addr:0x%p, size:0x%x\n",
464 memdesc->hostptr, memdesc->size);
465 return -ENOMEM;
466 }
467 }
468
469 return 0;
470}
471
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700472static void kgsl_coherent_free(struct kgsl_memdesc *memdesc)
473{
474 kgsl_driver.stats.coherent -= memdesc->size;
475 dma_free_coherent(NULL, memdesc->size,
476 memdesc->hostptr, memdesc->physaddr);
477}
478
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700479/* Global - also used by kgsl_drm.c */
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600480struct kgsl_memdesc_ops kgsl_page_alloc_ops = {
481 .free = kgsl_page_alloc_free,
482 .vmflags = kgsl_page_alloc_vmflags,
483 .vmfault = kgsl_page_alloc_vmfault,
484 .map_kernel_mem = kgsl_page_alloc_map_kernel,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700485};
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600486EXPORT_SYMBOL(kgsl_page_alloc_ops);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700487
488static struct kgsl_memdesc_ops kgsl_ebimem_ops = {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700489 .free = kgsl_ebimem_free,
490 .vmflags = kgsl_contiguous_vmflags,
491 .vmfault = kgsl_contiguous_vmfault,
Shubhraprakash Das387b6692012-08-11 18:26:29 -0700492 .map_kernel_mem = kgsl_ebimem_map_kernel,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700493};
494
495static struct kgsl_memdesc_ops kgsl_coherent_ops = {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700496 .free = kgsl_coherent_free,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700497};
498
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700499void kgsl_cache_range_op(struct kgsl_memdesc *memdesc, int op)
500{
Jordan Crousee9efb0b2013-05-28 16:54:19 -0600501 /*
502 * If the buffer is mapped in the kernel operate on that address
503 * otherwise use the user address
504 */
505
506 void *addr = (memdesc->hostptr) ?
507 memdesc->hostptr : (void *) memdesc->useraddr;
508
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700509 int size = memdesc->size;
510
Jeremy Gebbend1f8c902013-05-28 16:53:45 -0600511 if (addr != NULL) {
512 switch (op) {
513 case KGSL_CACHE_OP_FLUSH:
514 dmac_flush_range(addr, addr + size);
515 break;
516 case KGSL_CACHE_OP_CLEAN:
517 dmac_clean_range(addr, addr + size);
518 break;
519 case KGSL_CACHE_OP_INV:
520 dmac_inv_range(addr, addr + size);
521 break;
522 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700523 }
Jordan Croused17e9aa2011-10-12 16:57:48 -0600524 outer_cache_range_op_sg(memdesc->sg, memdesc->sglen, op);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700525}
526EXPORT_SYMBOL(kgsl_cache_range_op);
527
528static int
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600529_kgsl_sharedmem_page_alloc(struct kgsl_memdesc *memdesc,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700530 struct kgsl_pagetable *pagetable,
Jeremy Gebbena46f4272013-05-28 16:54:09 -0600531 size_t size)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700532{
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800533 int pcount = 0, order, ret = 0;
534 int j, len, page_size, sglen_alloc, sglen = 0;
Jordan Crouse89bd3232012-07-02 17:50:15 -0600535 struct page **pages = NULL;
536 pgprot_t page_prot = pgprot_writecombine(PAGE_KERNEL);
537 void *ptr;
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800538 unsigned int align;
Jordan Crouse2151cb92012-08-21 14:02:59 -0600539
Jordan Crousedc67dfb2012-10-25 09:41:46 -0600540 align = (memdesc->flags & KGSL_MEMALIGN_MASK) >> KGSL_MEMALIGN_SHIFT;
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800541
542 page_size = (align >= ilog2(SZ_64K) && size >= SZ_64K)
543 ? SZ_64K : PAGE_SIZE;
Jordan Crousedc67dfb2012-10-25 09:41:46 -0600544 /* update align flags for what we actually use */
Jeremy Gebbenfec05c22013-05-28 16:59:29 -0600545 if (page_size != PAGE_SIZE)
546 kgsl_memdesc_set_align(memdesc, ilog2(page_size));
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800547
548 /*
549 * There needs to be enough room in the sg structure to be able to
550 * service the allocation entirely with PAGE_SIZE sized chunks
551 */
552
553 sglen_alloc = PAGE_ALIGN(size) >> PAGE_SHIFT;
554
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700555 memdesc->pagetable = pagetable;
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600556 memdesc->ops = &kgsl_page_alloc_ops;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700557
Jordan Crouse100e4b32013-03-22 11:57:54 -0600558 memdesc->sglen_alloc = sglen_alloc;
559 memdesc->sg = kgsl_sg_alloc(memdesc->sglen_alloc);
Jordan Crousea652a072012-04-06 16:26:33 -0600560
Jordan Croused17e9aa2011-10-12 16:57:48 -0600561 if (memdesc->sg == NULL) {
562 ret = -ENOMEM;
563 goto done;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700564 }
565
Jordan Crouse89bd3232012-07-02 17:50:15 -0600566 /*
567 * Allocate space to store the list of pages to send to vmap.
Jordan Crousea0782b52013-07-18 16:20:08 -0600568 * This is an array of pointers so we can track 1024 pages per page
569 * of allocation. Since allocations can be as large as the user dares,
570 * we have to use the kmalloc/vmalloc trick here to make sure we can
571 * get the memory we need.
Jordan Crouse89bd3232012-07-02 17:50:15 -0600572 */
573
Jordan Crousea0782b52013-07-18 16:20:08 -0600574 if ((memdesc->sglen_alloc * sizeof(struct page *)) > PAGE_SIZE)
575 pages = vmalloc(memdesc->sglen_alloc * sizeof(struct page *));
576 else
577 pages = kmalloc(PAGE_SIZE, GFP_KERNEL);
Jordan Crouse89bd3232012-07-02 17:50:15 -0600578
579 if (pages == NULL) {
Jordan Crouse89bd3232012-07-02 17:50:15 -0600580 ret = -ENOMEM;
581 goto done;
582 }
583
Anshuman Danieecd5202012-02-17 19:52:49 +0530584 kmemleak_not_leak(memdesc->sg);
585
Jordan Crouse100e4b32013-03-22 11:57:54 -0600586 sg_init_table(memdesc->sg, memdesc->sglen_alloc);
Jordan Croused17e9aa2011-10-12 16:57:48 -0600587
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800588 len = size;
Jordan Crouse89bd3232012-07-02 17:50:15 -0600589
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800590 while (len > 0) {
591 struct page *page;
Jordan Crouse7ec25d82013-04-05 16:22:59 -0600592 unsigned int gfp_mask = __GFP_HIGHMEM;
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800593 int j;
Jordan Crouse89bd3232012-07-02 17:50:15 -0600594
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800595 /* don't waste space at the end of the allocation*/
596 if (len < page_size)
597 page_size = PAGE_SIZE;
598
Jordan Crouse7ec25d82013-04-05 16:22:59 -0600599 /*
600 * Don't do some of the more aggressive memory recovery
601 * techniques for large order allocations
602 */
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800603 if (page_size != PAGE_SIZE)
Jordan Crouse7ec25d82013-04-05 16:22:59 -0600604 gfp_mask |= __GFP_COMP | __GFP_NORETRY |
605 __GFP_NO_KSWAPD | __GFP_NOWARN;
606 else
Olav Haugan6f539842013-05-17 13:35:23 -0700607 gfp_mask |= GFP_KERNEL;
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800608
609 page = alloc_pages(gfp_mask, get_order(page_size));
610
611 if (page == NULL) {
612 if (page_size != PAGE_SIZE) {
613 page_size = PAGE_SIZE;
614 continue;
615 }
Jordan Crouse97c74d62012-11-01 12:42:42 -0600616
Hareesh Gunduff91aba2014-01-03 11:59:32 +0530617 /*
618 * Update sglen and memdesc size,as requested allocation
619 * not served fully. So that they can be correctly freed
620 * in kgsl_sharedmem_free().
621 */
622 memdesc->sglen = sglen;
623 memdesc->size = (size - len);
624
Jordan Crouse97c74d62012-11-01 12:42:42 -0600625 KGSL_CORE_ERR(
626 "Out of memory: only allocated %dKB of %dKB requested\n",
627 (size - len) >> 10, size >> 10);
628
629 ret = -ENOMEM;
630 goto done;
Jordan Croused17e9aa2011-10-12 16:57:48 -0600631 }
Jordan Crouse89bd3232012-07-02 17:50:15 -0600632
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800633 for (j = 0; j < page_size >> PAGE_SHIFT; j++)
634 pages[pcount++] = nth_page(page, j);
635
636 sg_set_page(&memdesc->sg[sglen++], page, page_size, 0);
637 len -= page_size;
Jordan Croused17e9aa2011-10-12 16:57:48 -0600638 }
Jordan Crouse7d3139b2012-05-18 10:05:02 -0600639
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800640 memdesc->sglen = sglen;
Hareesh Gunduff91aba2014-01-03 11:59:32 +0530641 memdesc->size = size;
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800642
Jordan Crouse89bd3232012-07-02 17:50:15 -0600643 /*
644 * All memory that goes to the user has to be zeroed out before it gets
645 * exposed to userspace. This means that the memory has to be mapped in
646 * the kernel, zeroed (memset) and then unmapped. This also means that
647 * the dcache has to be flushed to ensure coherency between the kernel
648 * and user pages. We used to pass __GFP_ZERO to alloc_page which mapped
649 * zeroed and unmaped each individual page, and then we had to turn
650 * around and call flush_dcache_page() on that page to clear the caches.
651 * This was killing us for performance. Instead, we found it is much
652 * faster to allocate the pages without GFP_ZERO, map the entire range,
653 * memset it, flush the range and then unmap - this results in a factor
654 * of 4 improvement for speed for large buffers. There is a small
655 * increase in speed for small buffers, but only on the order of a few
656 * microseconds at best. The only downside is that there needs to be
657 * enough temporary space in vmalloc to accomodate the map. This
658 * shouldn't be a problem, but if it happens, fall back to a much slower
659 * path
660 */
661
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800662 ptr = vmap(pages, pcount, VM_IOREMAP, page_prot);
Jordan Crouse89bd3232012-07-02 17:50:15 -0600663
664 if (ptr != NULL) {
665 memset(ptr, 0, memdesc->size);
666 dmac_flush_range(ptr, ptr + memdesc->size);
667 vunmap(ptr);
668 } else {
Jordan Crouse89bd3232012-07-02 17:50:15 -0600669 /* Very, very, very slow path */
670
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800671 for (j = 0; j < pcount; j++) {
Jordan Crouse89bd3232012-07-02 17:50:15 -0600672 ptr = kmap_atomic(pages[j]);
673 memset(ptr, 0, PAGE_SIZE);
674 dmac_flush_range(ptr, ptr + PAGE_SIZE);
675 kunmap_atomic(ptr);
676 }
677 }
678
Jeremy Gebben7018a212012-04-11 10:23:52 -0600679 outer_cache_range_op_sg(memdesc->sg, memdesc->sglen,
680 KGSL_CACHE_OP_FLUSH);
Jordan Croused17e9aa2011-10-12 16:57:48 -0600681
Jordan Croused17e9aa2011-10-12 16:57:48 -0600682 order = get_order(size);
683
684 if (order < 16)
685 kgsl_driver.stats.histogram[order]++;
686
687done:
Hareesh Gunduff91aba2014-01-03 11:59:32 +0530688 KGSL_STATS_ADD(memdesc->size, kgsl_driver.stats.page_alloc,
689 kgsl_driver.stats.page_alloc_max);
690
Jordan Crousea0782b52013-07-18 16:20:08 -0600691 if ((memdesc->sglen_alloc * sizeof(struct page *)) > PAGE_SIZE)
692 vfree(pages);
693 else
694 kfree(pages);
Jordan Crouse89bd3232012-07-02 17:50:15 -0600695
Jordan Croused17e9aa2011-10-12 16:57:48 -0600696 if (ret)
697 kgsl_sharedmem_free(memdesc);
698
699 return ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700700}
701
702int
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600703kgsl_sharedmem_page_alloc(struct kgsl_memdesc *memdesc,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700704 struct kgsl_pagetable *pagetable, size_t size)
705{
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600706 int ret = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700707 BUG_ON(size == 0);
708
709 size = ALIGN(size, PAGE_SIZE * 2);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700710
Jeremy Gebbena46f4272013-05-28 16:54:09 -0600711 ret = _kgsl_sharedmem_page_alloc(memdesc, pagetable, size);
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600712 if (!ret)
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600713 ret = kgsl_page_alloc_map_kernel(memdesc);
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600714 if (ret)
715 kgsl_sharedmem_free(memdesc);
716 return ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700717}
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600718EXPORT_SYMBOL(kgsl_sharedmem_page_alloc);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700719
720int
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600721kgsl_sharedmem_page_alloc_user(struct kgsl_memdesc *memdesc,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700722 struct kgsl_pagetable *pagetable,
Jordan Crousedc67dfb2012-10-25 09:41:46 -0600723 size_t size)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700724{
Jeremy Gebbena46f4272013-05-28 16:54:09 -0600725 return _kgsl_sharedmem_page_alloc(memdesc, pagetable, PAGE_ALIGN(size));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700726}
Harsh Vardhan Dwivedif99c2632012-03-15 14:17:11 -0600727EXPORT_SYMBOL(kgsl_sharedmem_page_alloc_user);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700728
729int
730kgsl_sharedmem_alloc_coherent(struct kgsl_memdesc *memdesc, size_t size)
731{
Jordan Croused17e9aa2011-10-12 16:57:48 -0600732 int result = 0;
733
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700734 size = ALIGN(size, PAGE_SIZE);
735
Jordan Croused17e9aa2011-10-12 16:57:48 -0600736 memdesc->size = size;
737 memdesc->ops = &kgsl_coherent_ops;
738
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700739 memdesc->hostptr = dma_alloc_coherent(NULL, size, &memdesc->physaddr,
740 GFP_KERNEL);
741 if (memdesc->hostptr == NULL) {
742 KGSL_CORE_ERR("dma_alloc_coherent(%d) failed\n", size);
Jordan Croused17e9aa2011-10-12 16:57:48 -0600743 result = -ENOMEM;
744 goto err;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700745 }
746
Jordan Croused17e9aa2011-10-12 16:57:48 -0600747 result = memdesc_sg_phys(memdesc, memdesc->physaddr, size);
748 if (result)
749 goto err;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700750
751 /* Record statistics */
752
753 KGSL_STATS_ADD(size, kgsl_driver.stats.coherent,
754 kgsl_driver.stats.coherent_max);
755
Jordan Croused17e9aa2011-10-12 16:57:48 -0600756err:
757 if (result)
758 kgsl_sharedmem_free(memdesc);
759
760 return result;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700761}
762EXPORT_SYMBOL(kgsl_sharedmem_alloc_coherent);
763
764void kgsl_sharedmem_free(struct kgsl_memdesc *memdesc)
765{
766 if (memdesc == NULL || memdesc->size == 0)
767 return;
768
Shubhraprakash Dasf402aeb2013-06-20 13:53:24 -0600769 if (memdesc->gpuaddr) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700770 kgsl_mmu_unmap(memdesc->pagetable, memdesc);
Shubhraprakash Dasf402aeb2013-06-20 13:53:24 -0600771 kgsl_mmu_put_gpuaddr(memdesc->pagetable, memdesc);
772 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700773
Jordan Croused17e9aa2011-10-12 16:57:48 -0600774 if (memdesc->ops && memdesc->ops->free)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700775 memdesc->ops->free(memdesc);
776
Rajeev Kulkarni8dfdc3362012-11-22 00:22:32 -0800777 kgsl_sg_free(memdesc->sg, memdesc->sglen_alloc);
Jordan Croused17e9aa2011-10-12 16:57:48 -0600778
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700779 memset(memdesc, 0, sizeof(*memdesc));
780}
781EXPORT_SYMBOL(kgsl_sharedmem_free);
782
783static int
784_kgsl_sharedmem_ebimem(struct kgsl_memdesc *memdesc,
785 struct kgsl_pagetable *pagetable, size_t size)
786{
Jordan Croused17e9aa2011-10-12 16:57:48 -0600787 int result = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700788
Jordan Croused17e9aa2011-10-12 16:57:48 -0600789 memdesc->size = size;
790 memdesc->pagetable = pagetable;
791 memdesc->ops = &kgsl_ebimem_ops;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700792 memdesc->physaddr = allocate_contiguous_ebi_nomap(size, SZ_8K);
793
794 if (memdesc->physaddr == 0) {
795 KGSL_CORE_ERR("allocate_contiguous_ebi_nomap(%d) failed\n",
796 size);
797 return -ENOMEM;
798 }
799
Jordan Croused17e9aa2011-10-12 16:57:48 -0600800 result = memdesc_sg_phys(memdesc, memdesc->physaddr, size);
801
802 if (result)
803 goto err;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700804
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700805 KGSL_STATS_ADD(size, kgsl_driver.stats.coherent,
806 kgsl_driver.stats.coherent_max);
807
Jordan Croused17e9aa2011-10-12 16:57:48 -0600808err:
809 if (result)
810 kgsl_sharedmem_free(memdesc);
811
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700812 return result;
813}
814
815int
816kgsl_sharedmem_ebimem_user(struct kgsl_memdesc *memdesc,
817 struct kgsl_pagetable *pagetable,
Jordan Crousedc67dfb2012-10-25 09:41:46 -0600818 size_t size)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700819{
820 size = ALIGN(size, PAGE_SIZE);
821 return _kgsl_sharedmem_ebimem(memdesc, pagetable, size);
822}
823EXPORT_SYMBOL(kgsl_sharedmem_ebimem_user);
824
825int
826kgsl_sharedmem_ebimem(struct kgsl_memdesc *memdesc,
827 struct kgsl_pagetable *pagetable, size_t size)
828{
829 int result;
830 size = ALIGN(size, 8192);
831 result = _kgsl_sharedmem_ebimem(memdesc, pagetable, size);
832
833 if (result)
834 return result;
835
836 memdesc->hostptr = ioremap(memdesc->physaddr, size);
837
838 if (memdesc->hostptr == NULL) {
839 KGSL_CORE_ERR("ioremap failed\n");
840 kgsl_sharedmem_free(memdesc);
841 return -ENOMEM;
842 }
843
844 return 0;
845}
846EXPORT_SYMBOL(kgsl_sharedmem_ebimem);
847
848int
849kgsl_sharedmem_readl(const struct kgsl_memdesc *memdesc,
850 uint32_t *dst,
851 unsigned int offsetbytes)
852{
Jeremy Gebbenaba13272012-01-31 17:31:23 -0700853 uint32_t *src;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700854 BUG_ON(memdesc == NULL || memdesc->hostptr == NULL || dst == NULL);
Jeremy Gebbenaba13272012-01-31 17:31:23 -0700855 WARN_ON(offsetbytes % sizeof(uint32_t) != 0);
856 if (offsetbytes % sizeof(uint32_t) != 0)
857 return -EINVAL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700858
Jeremy Gebbenaba13272012-01-31 17:31:23 -0700859 WARN_ON(offsetbytes + sizeof(uint32_t) > memdesc->size);
860 if (offsetbytes + sizeof(uint32_t) > memdesc->size)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700861 return -ERANGE;
Jeremy Gebbenaba13272012-01-31 17:31:23 -0700862 src = (uint32_t *)(memdesc->hostptr + offsetbytes);
863 *dst = *src;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700864 return 0;
865}
866EXPORT_SYMBOL(kgsl_sharedmem_readl);
867
868int
869kgsl_sharedmem_writel(const struct kgsl_memdesc *memdesc,
870 unsigned int offsetbytes,
871 uint32_t src)
872{
Jeremy Gebbenaba13272012-01-31 17:31:23 -0700873 uint32_t *dst;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700874 BUG_ON(memdesc == NULL || memdesc->hostptr == NULL);
Jeremy Gebbenaba13272012-01-31 17:31:23 -0700875 WARN_ON(offsetbytes % sizeof(uint32_t) != 0);
876 if (offsetbytes % sizeof(uint32_t) != 0)
877 return -EINVAL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700878
Jeremy Gebbenaba13272012-01-31 17:31:23 -0700879 WARN_ON(offsetbytes + sizeof(uint32_t) > memdesc->size);
880 if (offsetbytes + sizeof(uint32_t) > memdesc->size)
881 return -ERANGE;
Jeremy Gebbena3d07a42011-10-17 12:08:16 -0600882 kgsl_cffdump_setmem(memdesc->gpuaddr + offsetbytes,
Jeremy Gebbenaba13272012-01-31 17:31:23 -0700883 src, sizeof(uint32_t));
884 dst = (uint32_t *)(memdesc->hostptr + offsetbytes);
885 *dst = src;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700886 return 0;
887}
888EXPORT_SYMBOL(kgsl_sharedmem_writel);
889
890int
891kgsl_sharedmem_set(const struct kgsl_memdesc *memdesc, unsigned int offsetbytes,
892 unsigned int value, unsigned int sizebytes)
893{
894 BUG_ON(memdesc == NULL || memdesc->hostptr == NULL);
895 BUG_ON(offsetbytes + sizebytes > memdesc->size);
896
Jeremy Gebbena3d07a42011-10-17 12:08:16 -0600897 kgsl_cffdump_setmem(memdesc->gpuaddr + offsetbytes, value,
898 sizebytes);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700899 memset(memdesc->hostptr + offsetbytes, value, sizebytes);
900 return 0;
901}
902EXPORT_SYMBOL(kgsl_sharedmem_set);
Harsh Vardhan Dwivedi8cb835b2012-03-29 17:23:11 -0600903
904/*
905 * kgsl_sharedmem_map_vma - Map a user vma to physical memory
906 *
907 * @vma - The user vma to map
908 * @memdesc - The memory descriptor which contains information about the
909 * physical memory
910 *
911 * Return: 0 on success else error code
912 */
913int
914kgsl_sharedmem_map_vma(struct vm_area_struct *vma,
915 const struct kgsl_memdesc *memdesc)
916{
917 unsigned long addr = vma->vm_start;
918 unsigned long size = vma->vm_end - vma->vm_start;
919 int ret, i = 0;
920
921 if (!memdesc->sg || (size != memdesc->size) ||
922 (memdesc->sglen != (size / PAGE_SIZE)))
923 return -EINVAL;
924
925 for (; addr < vma->vm_end; addr += PAGE_SIZE, i++) {
926 ret = vm_insert_page(vma, addr, sg_page(&memdesc->sg[i]));
927 if (ret)
928 return ret;
929 }
930 return 0;
931}
932EXPORT_SYMBOL(kgsl_sharedmem_map_vma);
Jeremy Gebben158a5c02012-09-24 14:27:25 -0600933
934static const char * const memtype_str[] = {
935 [KGSL_MEMTYPE_OBJECTANY] = "any(0)",
936 [KGSL_MEMTYPE_FRAMEBUFFER] = "framebuffer",
937 [KGSL_MEMTYPE_RENDERBUFFER] = "renderbuffer",
938 [KGSL_MEMTYPE_ARRAYBUFFER] = "arraybuffer",
939 [KGSL_MEMTYPE_ELEMENTARRAYBUFFER] = "elementarraybuffer",
940 [KGSL_MEMTYPE_VERTEXARRAYBUFFER] = "vertexarraybuffer",
941 [KGSL_MEMTYPE_TEXTURE] = "texture",
942 [KGSL_MEMTYPE_SURFACE] = "surface",
943 [KGSL_MEMTYPE_EGL_SURFACE] = "egl_surface",
944 [KGSL_MEMTYPE_GL] = "gl",
945 [KGSL_MEMTYPE_CL] = "cl",
946 [KGSL_MEMTYPE_CL_BUFFER_MAP] = "cl_buffer_map",
947 [KGSL_MEMTYPE_CL_BUFFER_NOMAP] = "cl_buffer_nomap",
948 [KGSL_MEMTYPE_CL_IMAGE_MAP] = "cl_image_map",
949 [KGSL_MEMTYPE_CL_IMAGE_NOMAP] = "cl_image_nomap",
950 [KGSL_MEMTYPE_CL_KERNEL_STACK] = "cl_kernel_stack",
951 [KGSL_MEMTYPE_COMMAND] = "command",
952 [KGSL_MEMTYPE_2D] = "2d",
953 [KGSL_MEMTYPE_EGL_IMAGE] = "egl_image",
954 [KGSL_MEMTYPE_EGL_SHADOW] = "egl_shadow",
955 [KGSL_MEMTYPE_MULTISAMPLE] = "egl_multisample",
956 /* KGSL_MEMTYPE_KERNEL handled below, to avoid huge array */
957};
958
959void kgsl_get_memory_usage(char *name, size_t name_size, unsigned int memflags)
960{
961 unsigned char type;
962
963 type = (memflags & KGSL_MEMTYPE_MASK) >> KGSL_MEMTYPE_SHIFT;
964 if (type == KGSL_MEMTYPE_KERNEL)
965 strlcpy(name, "kernel", name_size);
966 else if (type < ARRAY_SIZE(memtype_str) && memtype_str[type] != NULL)
967 strlcpy(name, memtype_str[type], name_size);
968 else
969 snprintf(name, name_size, "unknown(%3d)", type);
970}
971EXPORT_SYMBOL(kgsl_get_memory_usage);