|
@@ -119,7 +119,7 @@ static int intelfb_alloc(struct drm_fb_helper *helper,
|
|
|
{
|
|
|
struct intel_fbdev *ifbdev =
|
|
|
container_of(helper, struct intel_fbdev, helper);
|
|
|
- struct drm_framebuffer *fb;
|
|
|
+ struct drm_framebuffer *fb = NULL;
|
|
|
struct drm_device *dev = helper->dev;
|
|
|
struct drm_i915_private *dev_priv = to_i915(dev);
|
|
|
struct drm_mode_fb_cmd2 mode_cmd = {};
|
|
@@ -138,6 +138,8 @@ static int intelfb_alloc(struct drm_fb_helper *helper,
|
|
|
mode_cmd.pixel_format = drm_mode_legacy_fb_format(sizes->surface_bpp,
|
|
|
sizes->surface_depth);
|
|
|
|
|
|
+ mutex_lock(&dev->struct_mutex);
|
|
|
+
|
|
|
size = mode_cmd.pitches[0] * mode_cmd.height;
|
|
|
size = PAGE_ALIGN(size);
|
|
|
|
|
@@ -165,16 +167,19 @@ static int intelfb_alloc(struct drm_fb_helper *helper,
|
|
|
ret = intel_pin_and_fence_fb_obj(NULL, fb, NULL);
|
|
|
if (ret) {
|
|
|
DRM_ERROR("failed to pin obj: %d\n", ret);
|
|
|
- goto out_fb;
|
|
|
+ goto out;
|
|
|
}
|
|
|
|
|
|
+ mutex_unlock(&dev->struct_mutex);
|
|
|
+
|
|
|
ifbdev->fb = to_intel_framebuffer(fb);
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
-out_fb:
|
|
|
- drm_framebuffer_remove(fb);
|
|
|
out:
|
|
|
+ mutex_unlock(&dev->struct_mutex);
|
|
|
+ if (!IS_ERR_OR_NULL(fb))
|
|
|
+ drm_framebuffer_unreference(fb);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
@@ -192,8 +197,6 @@ static int intelfb_create(struct drm_fb_helper *helper,
|
|
|
int size, ret;
|
|
|
bool prealloc = false;
|
|
|
|
|
|
- mutex_lock(&dev->struct_mutex);
|
|
|
-
|
|
|
if (intel_fb &&
|
|
|
(sizes->fb_width > intel_fb->base.width ||
|
|
|
sizes->fb_height > intel_fb->base.height)) {
|
|
@@ -208,7 +211,7 @@ static int intelfb_create(struct drm_fb_helper *helper,
|
|
|
DRM_DEBUG_KMS("no BIOS fb, allocating a new one\n");
|
|
|
ret = intelfb_alloc(helper, sizes);
|
|
|
if (ret)
|
|
|
- goto out_unlock;
|
|
|
+ return ret;
|
|
|
intel_fb = ifbdev->fb;
|
|
|
} else {
|
|
|
DRM_DEBUG_KMS("re-using BIOS fb\n");
|
|
@@ -220,6 +223,8 @@ static int intelfb_create(struct drm_fb_helper *helper,
|
|
|
obj = intel_fb->obj;
|
|
|
size = obj->base.size;
|
|
|
|
|
|
+ mutex_lock(&dev->struct_mutex);
|
|
|
+
|
|
|
info = drm_fb_helper_alloc_fbi(helper);
|
|
|
if (IS_ERR(info)) {
|
|
|
ret = PTR_ERR(info);
|
|
@@ -281,7 +286,6 @@ out_destroy_fbi:
|
|
|
out_unpin:
|
|
|
i915_gem_object_ggtt_unpin(obj);
|
|
|
drm_gem_object_unreference(&obj->base);
|
|
|
-out_unlock:
|
|
|
mutex_unlock(&dev->struct_mutex);
|
|
|
return ret;
|
|
|
}
|