static void tegra_fbdev_fb_destroy(struct fb_info *info)
{
struct drm_fb_helper *helper = info->par;
- struct drm_framebuffer *fb = helper->fb;
- struct tegra_bo *bo = tegra_fb_get_plane(fb, 0);
+ struct tegra_bo *bo = tegra_fb_get_plane(helper->fb, 0);
drm_fb_helper_fini(helper);
vunmap(bo->vaddr);
bo->vaddr = NULL;
}
- drm_framebuffer_remove(fb);
+ drm_client_buffer_delete(helper->buffer);
drm_client_release(&helper->client);
}
int tegra_fbdev_driver_fbdev_probe(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes)
{
- struct tegra_drm *tegra = helper->dev->dev_private;
- struct drm_device *drm = helper->dev;
- struct drm_mode_fb_cmd2 cmd = { 0 };
+ struct drm_client_dev *client = &helper->client;
+ struct drm_device *drm = client->dev;
+ struct drm_file *file = client->file;
+ struct tegra_drm *tegra = drm->dev_private;
struct fb_info *info = helper->info;
u32 fourcc, pitch;
u64 size;
const struct drm_format_info *format;
- struct drm_framebuffer *fb;
struct tegra_bo *bo;
+ struct drm_gem_object *gem;
+ u32 handle;
+ struct drm_client_buffer *buffer;
int err;
fourcc = drm_mode_legacy_fb_format(sizes->surface_bpp, sizes->surface_depth);
bo = tegra_bo_create(drm, size, 0);
if (IS_ERR(bo))
return PTR_ERR(bo);
+ gem = &bo->gem;
- cmd.pixel_format = fourcc;
- cmd.width = sizes->surface_width;
- cmd.height = sizes->surface_height;
- cmd.pitches[0] = pitch;
-
- fb = tegra_fb_alloc(drm, format, &cmd, &bo, 1);
- if (IS_ERR(fb)) {
- err = PTR_ERR(fb);
- dev_err(drm->dev, "failed to allocate DRM framebuffer: %d\n",
- err);
- drm_gem_object_put(&bo->gem);
- return PTR_ERR(fb);
+ err = drm_gem_handle_create(file, gem, &handle);
+ if (err)
+ goto err_drm_gem_object_put;
+
+ buffer = drm_client_buffer_create(client, sizes->surface_width, sizes->surface_height,
+ fourcc, handle, pitch);
+ if (IS_ERR(buffer)) {
+ err = PTR_ERR(buffer);
+ goto err_drm_gem_handle_delete;
}
helper->funcs = &tegra_fbdev_helper_funcs;
- helper->fb = fb;
+ helper->buffer = buffer;
+ helper->fb = buffer->fb;
info->fbops = &tegra_fb_ops;
if (!bo->vaddr) {
dev_err(drm->dev, "failed to vmap() framebuffer\n");
err = -ENOMEM;
- goto destroy;
+ goto err_drm_client_buffer_delete;
}
}
info->flags |= FBINFO_VIRTFB;
info->screen_buffer = bo->vaddr;
- info->screen_size = bo->gem.size;
+ info->screen_size = gem->size;
info->fix.smem_start = (unsigned long)(bo->iova);
- info->fix.smem_len = bo->gem.size;
+ info->fix.smem_len = gem->size;
+
+ /* The handle is only needed for creating the framebuffer. */
+ drm_gem_handle_delete(file, handle);
+
+ /* The framebuffer still holds a reference on the GEM object. */
+ drm_gem_object_put(gem);
return 0;
-destroy:
- drm_framebuffer_remove(fb);
+err_drm_client_buffer_delete:
+ drm_client_buffer_delete(buffer);
+err_drm_gem_handle_delete:
+ drm_gem_handle_delete(file, handle);
+err_drm_gem_object_put:
+ drm_gem_object_put(gem);
return err;
}