gma500: Ensure the frame buffer has a linear virtual mapping
We need this for the framebuffer in order to ensure that the kernel framebuffer layer can handle it when using KMS. Except for the base framebuffer this isn't a concern. Add an npage field to the gtt as too many copies of the page calculation are getting spread around the code. Signed-off-by: Alan Cox <alan@linux.intel.com> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
This commit is contained in:
parent
0c9e98af5e
commit
9460e84a91
|
@ -254,17 +254,13 @@ static int psbfb_mmap(struct fb_info *info, struct vm_area_struct *vma)
|
||||||
vma->vm_pgoff, fb_screen_base,
|
vma->vm_pgoff, fb_screen_base,
|
||||||
dev_priv->vram_addr);
|
dev_priv->vram_addr);
|
||||||
|
|
||||||
/* FIXME: ultimately this needs to become 'if entirely stolen memory' */
|
/* If this is a GEM object then info->screen_base is the virtual
|
||||||
if (1 || fb_screen_base == dev_priv->vram_addr) {
|
kernel remapping of the object. FIXME: Review if this is
|
||||||
vma->vm_ops = &psbfb_vm_ops;
|
suitable for our mmap work */
|
||||||
vma->vm_private_data = (void *)psbfb;
|
vma->vm_ops = &psbfb_vm_ops;
|
||||||
vma->vm_flags |= VM_RESERVED | VM_IO |
|
vma->vm_private_data = (void *)psbfb;
|
||||||
VM_MIXEDMAP | VM_DONTEXPAND;
|
vma->vm_flags |= VM_RESERVED | VM_IO |
|
||||||
} else {
|
VM_MIXEDMAP | VM_DONTEXPAND;
|
||||||
/* GTT memory backed by kernel/user pages, needs a different
|
|
||||||
approach ? - GEM ? */
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -349,8 +345,6 @@ err:
|
||||||
*
|
*
|
||||||
* FIXME: console speed up - allocate twice the space if room and use
|
* FIXME: console speed up - allocate twice the space if room and use
|
||||||
* hardware scrolling for acceleration.
|
* hardware scrolling for acceleration.
|
||||||
* FIXME: we need to vm_map_ram a linear mapping if the object has to
|
|
||||||
* be GEM host mapped, otherwise the cfb layer's brain will fall out.
|
|
||||||
*/
|
*/
|
||||||
static struct gtt_range *psbfb_alloc(struct drm_device *dev, int aligned_size)
|
static struct gtt_range *psbfb_alloc(struct drm_device *dev, int aligned_size)
|
||||||
{
|
{
|
||||||
|
@ -439,10 +433,22 @@ static int psbfb_create(struct psb_fbdev *fbdev,
|
||||||
info->fix.smem_start = dev->mode_config.fb_base;
|
info->fix.smem_start = dev->mode_config.fb_base;
|
||||||
info->fix.smem_len = size;
|
info->fix.smem_len = size;
|
||||||
|
|
||||||
/* Accessed via stolen memory directly, This only works for stolem
|
if (backing->stolen) {
|
||||||
memory however. Need to address this once we start using gtt
|
/* Accessed stolen memory directly */
|
||||||
pages we allocate. FIXME: vm_map_ram for that case */
|
info->screen_base = (char *)dev_priv->vram_addr +
|
||||||
info->screen_base = (char *)dev_priv->vram_addr + backing->offset;
|
backing->offset;
|
||||||
|
} else {
|
||||||
|
/* Pin the pages into the GTT and create a mapping to them */
|
||||||
|
psb_gtt_pin(backing);
|
||||||
|
info->screen_base = vm_map_ram(backing->pages, backing->npage,
|
||||||
|
-1, PAGE_KERNEL);
|
||||||
|
if (info->screen_base == NULL) {
|
||||||
|
psb_gtt_unpin(backing);
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto out_err0;
|
||||||
|
}
|
||||||
|
psbfb->vm_map = 1;
|
||||||
|
}
|
||||||
info->screen_size = size;
|
info->screen_size = size;
|
||||||
memset(info->screen_base, 0, size);
|
memset(info->screen_base, 0, size);
|
||||||
|
|
||||||
|
@ -570,6 +576,13 @@ int psb_fbdev_destroy(struct drm_device *dev, struct psb_fbdev *fbdev)
|
||||||
|
|
||||||
if (fbdev->psb_fb_helper.fbdev) {
|
if (fbdev->psb_fb_helper.fbdev) {
|
||||||
info = fbdev->psb_fb_helper.fbdev;
|
info = fbdev->psb_fb_helper.fbdev;
|
||||||
|
|
||||||
|
/* If this is our base framebuffer then kill any virtual map
|
||||||
|
for the framebuffer layer and unpin it */
|
||||||
|
if (psbfb->vm_map) {
|
||||||
|
vm_unmap_ram(info->screen_base, psbfb->gtt->npage);
|
||||||
|
psb_gtt_unpin(psbfb->gtt);
|
||||||
|
}
|
||||||
/* FIXME: this is a bit more inside knowledge than I'd like
|
/* FIXME: this is a bit more inside knowledge than I'd like
|
||||||
but I don't see how to make a fake GEM object of the
|
but I don't see how to make a fake GEM object of the
|
||||||
stolen space nicely */
|
stolen space nicely */
|
||||||
|
|
|
@ -33,6 +33,7 @@ struct psb_framebuffer {
|
||||||
struct address_space *addr_space;
|
struct address_space *addr_space;
|
||||||
struct fb_info *fbdev;
|
struct fb_info *fbdev;
|
||||||
struct gtt_range *gtt;
|
struct gtt_range *gtt;
|
||||||
|
bool vm_map; /* True if we must undo a vm_map_ram */
|
||||||
};
|
};
|
||||||
|
|
||||||
struct psb_fbdev {
|
struct psb_fbdev {
|
||||||
|
|
|
@ -79,7 +79,6 @@ u32 *psb_gtt_entry(struct drm_device *dev, struct gtt_range *r)
|
||||||
static int psb_gtt_insert(struct drm_device *dev, struct gtt_range *r)
|
static int psb_gtt_insert(struct drm_device *dev, struct gtt_range *r)
|
||||||
{
|
{
|
||||||
u32 *gtt_slot, pte;
|
u32 *gtt_slot, pte;
|
||||||
int numpages = (r->resource.end + 1 - r->resource.start) >> PAGE_SHIFT;
|
|
||||||
struct page **pages;
|
struct page **pages;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
@ -94,10 +93,10 @@ static int psb_gtt_insert(struct drm_device *dev, struct gtt_range *r)
|
||||||
pages = r->pages;
|
pages = r->pages;
|
||||||
|
|
||||||
/* Make sure changes are visible to the GPU */
|
/* Make sure changes are visible to the GPU */
|
||||||
set_pages_array_uc(pages, numpages);
|
set_pages_array_uc(pages, r->npage);
|
||||||
|
|
||||||
/* Write our page entries into the GTT itself */
|
/* Write our page entries into the GTT itself */
|
||||||
for (i = 0; i < numpages; i++) {
|
for (i = 0; i < r->npage; i++) {
|
||||||
pte = psb_gtt_mask_pte(page_to_pfn(*pages++), 0/*type*/);
|
pte = psb_gtt_mask_pte(page_to_pfn(*pages++), 0/*type*/);
|
||||||
iowrite32(pte, gtt_slot++);
|
iowrite32(pte, gtt_slot++);
|
||||||
}
|
}
|
||||||
|
@ -120,7 +119,6 @@ static void psb_gtt_remove(struct drm_device *dev, struct gtt_range *r)
|
||||||
{
|
{
|
||||||
struct drm_psb_private *dev_priv = dev->dev_private;
|
struct drm_psb_private *dev_priv = dev->dev_private;
|
||||||
u32 *gtt_slot, pte;
|
u32 *gtt_slot, pte;
|
||||||
int numpages = (r->resource.end + 1 - r->resource.start) >> PAGE_SHIFT;
|
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
WARN_ON(r->stolen);
|
WARN_ON(r->stolen);
|
||||||
|
@ -128,10 +126,10 @@ static void psb_gtt_remove(struct drm_device *dev, struct gtt_range *r)
|
||||||
gtt_slot = psb_gtt_entry(dev, r);
|
gtt_slot = psb_gtt_entry(dev, r);
|
||||||
pte = psb_gtt_mask_pte(page_to_pfn(dev_priv->scratch_page), 0);
|
pte = psb_gtt_mask_pte(page_to_pfn(dev_priv->scratch_page), 0);
|
||||||
|
|
||||||
for (i = 0; i < numpages; i++)
|
for (i = 0; i < r->npage; i++)
|
||||||
iowrite32(pte, gtt_slot++);
|
iowrite32(pte, gtt_slot++);
|
||||||
ioread32(gtt_slot - 1);
|
ioread32(gtt_slot - 1);
|
||||||
set_pages_array_wb(r->pages, numpages);
|
set_pages_array_wb(r->pages, r->npage);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -149,7 +147,7 @@ static int psb_gtt_attach_pages(struct gtt_range *gt)
|
||||||
struct address_space *mapping;
|
struct address_space *mapping;
|
||||||
int i;
|
int i;
|
||||||
struct page *p;
|
struct page *p;
|
||||||
int pages = (gt->resource.end + 1 - gt->resource.start) >> PAGE_SHIFT;
|
int pages = gt->gem.size / PAGE_SIZE;
|
||||||
|
|
||||||
WARN_ON(gt->pages);
|
WARN_ON(gt->pages);
|
||||||
|
|
||||||
|
@ -160,6 +158,8 @@ static int psb_gtt_attach_pages(struct gtt_range *gt)
|
||||||
gt->pages = kmalloc(pages * sizeof(struct page *), GFP_KERNEL);
|
gt->pages = kmalloc(pages * sizeof(struct page *), GFP_KERNEL);
|
||||||
if (gt->pages == NULL)
|
if (gt->pages == NULL)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
gt->npage = pages;
|
||||||
|
|
||||||
for (i = 0; i < pages; i++) {
|
for (i = 0; i < pages; i++) {
|
||||||
/* FIXME: review flags later */
|
/* FIXME: review flags later */
|
||||||
p = read_cache_page_gfp(mapping, i,
|
p = read_cache_page_gfp(mapping, i,
|
||||||
|
@ -191,9 +191,7 @@ err:
|
||||||
static void psb_gtt_detach_pages(struct gtt_range *gt)
|
static void psb_gtt_detach_pages(struct gtt_range *gt)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
int pages = (gt->resource.end + 1 - gt->resource.start) >> PAGE_SHIFT;
|
for (i = 0; i < gt->npage; i++) {
|
||||||
|
|
||||||
for (i = 0; i < pages; i++) {
|
|
||||||
/* FIXME: do we need to force dirty */
|
/* FIXME: do we need to force dirty */
|
||||||
set_page_dirty(gt->pages[i]);
|
set_page_dirty(gt->pages[i]);
|
||||||
/* Undo the reference we took when populating the table */
|
/* Undo the reference we took when populating the table */
|
||||||
|
|
|
@ -46,9 +46,10 @@ struct gtt_range {
|
||||||
struct kref kref;
|
struct kref kref;
|
||||||
struct drm_gem_object gem; /* GEM high level stuff */
|
struct drm_gem_object gem; /* GEM high level stuff */
|
||||||
int in_gart; /* Currently in the GART (ref ct) */
|
int in_gart; /* Currently in the GART (ref ct) */
|
||||||
bool stolen; /* Backed from stolen RAM */
|
bool stolen; /* Backed from stolen RAM */
|
||||||
bool mmapping; /* Is mmappable */
|
bool mmapping; /* Is mmappable */
|
||||||
struct page **pages; /* Backing pages if present */
|
struct page **pages; /* Backing pages if present */
|
||||||
|
int npage; /* Number of backing pages */
|
||||||
};
|
};
|
||||||
|
|
||||||
extern struct gtt_range *psb_gtt_alloc_range(struct drm_device *dev, int len,
|
extern struct gtt_range *psb_gtt_alloc_range(struct drm_device *dev, int len,
|
||||||
|
|
Loading…
Reference in New Issue