|
@@ -35,11 +35,18 @@
|
|
|
|
|
|
#include <drm/drmP.h>
|
|
#include <drm/drmP.h>
|
|
#include <linux/export.h>
|
|
#include <linux/export.h>
|
|
|
|
+#include <linux/seq_file.h>
|
|
#if defined(__ia64__)
|
|
#if defined(__ia64__)
|
|
#include <linux/efi.h>
|
|
#include <linux/efi.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/slab.h>
|
|
#endif
|
|
#endif
|
|
|
|
|
|
|
|
+struct drm_vma_entry {
|
|
|
|
+ struct list_head head;
|
|
|
|
+ struct vm_area_struct *vma;
|
|
|
|
+ pid_t pid;
|
|
|
|
+};
|
|
|
|
+
|
|
static void drm_vm_open(struct vm_area_struct *vma);
|
|
static void drm_vm_open(struct vm_area_struct *vma);
|
|
static void drm_vm_close(struct vm_area_struct *vma);
|
|
static void drm_vm_close(struct vm_area_struct *vma);
|
|
|
|
|
|
@@ -662,3 +669,72 @@ int drm_mmap(struct file *filp, struct vm_area_struct *vma)
|
|
return ret;
|
|
return ret;
|
|
}
|
|
}
|
|
EXPORT_SYMBOL(drm_mmap);
|
|
EXPORT_SYMBOL(drm_mmap);
|
|
|
|
+
|
|
|
|
+void drm_legacy_vma_flush(struct drm_device *dev)
|
|
|
|
+{
|
|
|
|
+ struct drm_vma_entry *vma, *vma_temp;
|
|
|
|
+
|
|
|
|
+ /* Clear vma list (only needed for legacy drivers) */
|
|
|
|
+ list_for_each_entry_safe(vma, vma_temp, &dev->vmalist, head) {
|
|
|
|
+ list_del(&vma->head);
|
|
|
|
+ kfree(vma);
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+#if DRM_DEBUG_CODE
|
|
|
|
+
|
|
|
|
+int drm_vma_info(struct seq_file *m, void *data)
|
|
|
|
+{
|
|
|
|
+ struct drm_info_node *node = (struct drm_info_node *) m->private;
|
|
|
|
+ struct drm_device *dev = node->minor->dev;
|
|
|
|
+ struct drm_vma_entry *pt;
|
|
|
|
+ struct vm_area_struct *vma;
|
|
|
|
+ unsigned long vma_count = 0;
|
|
|
|
+#if defined(__i386__)
|
|
|
|
+ unsigned int pgprot;
|
|
|
|
+#endif
|
|
|
|
+
|
|
|
|
+ mutex_lock(&dev->struct_mutex);
|
|
|
|
+ list_for_each_entry(pt, &dev->vmalist, head)
|
|
|
|
+ vma_count++;
|
|
|
|
+
|
|
|
|
+ seq_printf(m, "vma use count: %lu, high_memory = %pK, 0x%pK\n",
|
|
|
|
+ vma_count, high_memory,
|
|
|
|
+ (void *)(unsigned long)virt_to_phys(high_memory));
|
|
|
|
+
|
|
|
|
+ list_for_each_entry(pt, &dev->vmalist, head) {
|
|
|
|
+ vma = pt->vma;
|
|
|
|
+ if (!vma)
|
|
|
|
+ continue;
|
|
|
|
+ seq_printf(m,
|
|
|
|
+ "\n%5d 0x%pK-0x%pK %c%c%c%c%c%c 0x%08lx000",
|
|
|
|
+ pt->pid,
|
|
|
|
+ (void *)vma->vm_start, (void *)vma->vm_end,
|
|
|
|
+ vma->vm_flags & VM_READ ? 'r' : '-',
|
|
|
|
+ vma->vm_flags & VM_WRITE ? 'w' : '-',
|
|
|
|
+ vma->vm_flags & VM_EXEC ? 'x' : '-',
|
|
|
|
+ vma->vm_flags & VM_MAYSHARE ? 's' : 'p',
|
|
|
|
+ vma->vm_flags & VM_LOCKED ? 'l' : '-',
|
|
|
|
+ vma->vm_flags & VM_IO ? 'i' : '-',
|
|
|
|
+ vma->vm_pgoff);
|
|
|
|
+
|
|
|
|
+#if defined(__i386__)
|
|
|
|
+ pgprot = pgprot_val(vma->vm_page_prot);
|
|
|
|
+ seq_printf(m, " %c%c%c%c%c%c%c%c%c",
|
|
|
|
+ pgprot & _PAGE_PRESENT ? 'p' : '-',
|
|
|
|
+ pgprot & _PAGE_RW ? 'w' : 'r',
|
|
|
|
+ pgprot & _PAGE_USER ? 'u' : 's',
|
|
|
|
+ pgprot & _PAGE_PWT ? 't' : 'b',
|
|
|
|
+ pgprot & _PAGE_PCD ? 'u' : 'c',
|
|
|
|
+ pgprot & _PAGE_ACCESSED ? 'a' : '-',
|
|
|
|
+ pgprot & _PAGE_DIRTY ? 'd' : '-',
|
|
|
|
+ pgprot & _PAGE_PSE ? 'm' : 'k',
|
|
|
|
+ pgprot & _PAGE_GLOBAL ? 'g' : 'l');
|
|
|
|
+#endif
|
|
|
|
+ seq_printf(m, "\n");
|
|
|
|
+ }
|
|
|
|
+ mutex_unlock(&dev->struct_mutex);
|
|
|
|
+ return 0;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+#endif
|