Instead of receiving the num_crts as a parameter, we can read it
directly from the mode_config structure.  I audited the drivers that
invoke this helper and I believe all of them initialize the mode_config
struct accordingly, prior to calling the fb_helper.
I used the following coccinelle hack to make this transformation, except
for the function headers and comment updates.  The first and second
rules are split because I couldn't find a way to remove the unused
temporary variables at the same time I removed the parameter.
// <smpl>
@r@
expression A,B,D,E;
identifier C;
@@
(
- drm_fb_helper_init(A,B,C,D)
+ drm_fb_helper_init(A,B,D)
|
- drm_fbdev_cma_init_with_funcs(A,B,C,D,E)
+ drm_fbdev_cma_init_with_funcs(A,B,D,E)
|
- drm_fbdev_cma_init(A,B,C,D)
+ drm_fbdev_cma_init(A,B,D)
)
@@
expression A,B,C,D,E;
@@
(
- drm_fb_helper_init(A,B,C,D)
+ drm_fb_helper_init(A,B,D)
|
- drm_fbdev_cma_init_with_funcs(A,B,C,D,E)
+ drm_fbdev_cma_init_with_funcs(A,B,D,E)
|
- drm_fbdev_cma_init(A,B,C,D)
+ drm_fbdev_cma_init(A,B,D)
)
@@
identifier r.C;
type T;
expression V;
@@
- T C;
<...
when != C
- C = V;
...>
// </smpl>
Changes since v1:
 - Rebased on top of the tip of drm-misc-next.
 - Remove mention to sti since a proper fix got merged.
Suggested-by: Daniel Vetter <daniel.vetter@intel.com>
Signed-off-by: Gabriel Krisman Bertazi <krisman@collabora.co.uk>
Reviewed-by: Eric Anholt <eric@anholt.net>
Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
Link: http://patchwork.freedesktop.org/patch/msgid/20170202162640.27261-1-krisman@collabora.co.uk
                        &amdgpu_fb_helper_funcs);
 
        ret = drm_fb_helper_init(adev->ddev, &rfbdev->helper,
-                                adev->mode_info.num_crtc,
                                 AMDGPUFB_CONN_LIMIT);
        if (ret) {
                kfree(rfbdev);
 
        drm_kms_helper_poll_init(drm);
 
        arcpgu->fbdev = drm_fbdev_cma_init(drm, 16,
-                                             drm->mode_config.num_crtc,
-                                             drm->mode_config.num_connector);
+                                          drm->mode_config.num_connector);
        if (IS_ERR(arcpgu->fbdev)) {
                ret = PTR_ERR(arcpgu->fbdev);
                arcpgu->fbdev = NULL;
 
        drm_mode_config_reset(drm);
        drm_kms_helper_poll_init(drm);
 
-       hdlcd->fbdev = drm_fbdev_cma_init(drm, 32, drm->mode_config.num_crtc,
+       hdlcd->fbdev = drm_fbdev_cma_init(drm, 32,
                                          drm->mode_config.num_connector);
 
        if (IS_ERR(hdlcd->fbdev)) {
 
 
        drm_mode_config_reset(drm);
 
-       malidp->fbdev = drm_fbdev_cma_init(drm, 32, drm->mode_config.num_crtc,
+       malidp->fbdev = drm_fbdev_cma_init(drm, 32,
                                           drm->mode_config.num_connector);
 
        if (IS_ERR(malidp->fbdev)) {
 
 
        drm_fb_helper_prepare(dev, fbh, &armada_fb_helper_funcs);
 
-       ret = drm_fb_helper_init(dev, fbh, 1, 1);
+       ret = drm_fb_helper_init(dev, fbh, 1);
        if (ret) {
                DRM_ERROR("failed to initialize drm fb helper\n");
                goto err_fb_helper;
 
 
        drm_fb_helper_prepare(dev, &afbdev->helper, &ast_fb_helper_funcs);
 
-       ret = drm_fb_helper_init(dev, &afbdev->helper,
-                                1, 1);
+       ret = drm_fb_helper_init(dev, &afbdev->helper, 1);
        if (ret)
                goto free;
 
 
        platform_set_drvdata(pdev, dev);
 
        dc->fbdev = drm_fbdev_cma_init(dev, 24,
-                       dev->mode_config.num_crtc,
                        dev->mode_config.num_connector);
        if (IS_ERR(dc->fbdev))
                dc->fbdev = NULL;
 
        drm_fb_helper_prepare(bochs->dev, &bochs->fb.helper,
                              &bochs_fb_helper_funcs);
 
-       ret = drm_fb_helper_init(bochs->dev, &bochs->fb.helper,
-                                1, 1);
+       ret = drm_fb_helper_init(bochs->dev, &bochs->fb.helper, 1);
        if (ret)
                return ret;
 
 
                              &cirrus_fb_helper_funcs);
 
        ret = drm_fb_helper_init(cdev->dev, &gfbdev->helper,
-                                cdev->num_crtc, CIRRUSFB_CONN_LIMIT);
+                                CIRRUSFB_CONN_LIMIT);
        if (ret)
                return ret;
 
 
  * drm_fbdev_cma_init_with_funcs() - Allocate and initializes a drm_fbdev_cma struct
  * @dev: DRM device
  * @preferred_bpp: Preferred bits per pixel for the device
- * @num_crtc: Number of CRTCs
  * @max_conn_count: Maximum number of connectors
  * @funcs: fb helper functions, in particular a custom dirty() callback
  *
  * Returns a newly allocated drm_fbdev_cma struct or a ERR_PTR.
  */
 struct drm_fbdev_cma *drm_fbdev_cma_init_with_funcs(struct drm_device *dev,
-       unsigned int preferred_bpp, unsigned int num_crtc,
-       unsigned int max_conn_count, const struct drm_framebuffer_funcs *funcs)
+       unsigned int preferred_bpp, unsigned int max_conn_count,
+       const struct drm_framebuffer_funcs *funcs)
 {
        struct drm_fbdev_cma *fbdev_cma;
        struct drm_fb_helper *helper;
 
        drm_fb_helper_prepare(dev, helper, &drm_fb_cma_helper_funcs);
 
-       ret = drm_fb_helper_init(dev, helper, num_crtc, max_conn_count);
+       ret = drm_fb_helper_init(dev, helper, max_conn_count);
        if (ret < 0) {
                dev_err(dev->dev, "Failed to initialize drm fb helper.\n");
                goto err_free;
  * Returns a newly allocated drm_fbdev_cma struct or a ERR_PTR.
  */
 struct drm_fbdev_cma *drm_fbdev_cma_init(struct drm_device *dev,
-       unsigned int preferred_bpp, unsigned int num_crtc,
-       unsigned int max_conn_count)
+       unsigned int preferred_bpp, unsigned int max_conn_count)
 {
-       return drm_fbdev_cma_init_with_funcs(dev, preferred_bpp, num_crtc,
-                               max_conn_count, &drm_fb_cma_funcs);
+       return drm_fbdev_cma_init_with_funcs(dev, preferred_bpp,
+                                            max_conn_count,
+                                            &drm_fb_cma_funcs);
 }
 EXPORT_SYMBOL_GPL(drm_fbdev_cma_init);
 
 
  * drm_fb_helper_init - initialize a drm_fb_helper structure
  * @dev: drm device
  * @fb_helper: driver-allocated fbdev helper structure to initialize
- * @crtc_count: maximum number of crtcs to support in this fbdev emulation
  * @max_conn_count: max connector count
  *
  * This allocates the structures for the fbdev helper with the given limits.
  */
 int drm_fb_helper_init(struct drm_device *dev,
                       struct drm_fb_helper *fb_helper,
-                      int crtc_count, int max_conn_count)
+                      int max_conn_count)
 {
        struct drm_crtc *crtc;
+       struct drm_mode_config *config = &dev->mode_config;
        int i;
 
        if (!drm_fbdev_emulation)
        if (!max_conn_count)
                return -EINVAL;
 
-       fb_helper->crtc_info = kcalloc(crtc_count, sizeof(struct drm_fb_helper_crtc), GFP_KERNEL);
+       fb_helper->crtc_info = kcalloc(config->num_crtc, sizeof(struct drm_fb_helper_crtc), GFP_KERNEL);
        if (!fb_helper->crtc_info)
                return -ENOMEM;
 
-       fb_helper->crtc_count = crtc_count;
+       fb_helper->crtc_count = config->num_crtc;
        fb_helper->connector_info = kcalloc(dev->mode_config.num_connector, sizeof(struct drm_fb_helper_connector *), GFP_KERNEL);
        if (!fb_helper->connector_info) {
                kfree(fb_helper->crtc_info);
        fb_helper->connector_info_alloc_count = dev->mode_config.num_connector;
        fb_helper->connector_count = 0;
 
-       for (i = 0; i < crtc_count; i++) {
+       for (i = 0; i < fb_helper->crtc_count; i++) {
                fb_helper->crtc_info[i].mode_set.connectors =
                        kcalloc(max_conn_count,
                                sizeof(struct drm_connector *),
 
        struct exynos_drm_fbdev *fbdev;
        struct exynos_drm_private *private = dev->dev_private;
        struct drm_fb_helper *helper;
-       unsigned int num_crtc;
        int ret;
 
        if (!dev->mode_config.num_crtc || !dev->mode_config.num_connector)
 
        drm_fb_helper_prepare(dev, helper, &exynos_drm_fb_helper_funcs);
 
-       num_crtc = dev->mode_config.num_crtc;
-
-       ret = drm_fb_helper_init(dev, helper, num_crtc, MAX_CONNECTOR);
+       ret = drm_fb_helper_init(dev, helper, MAX_CONNECTOR);
        if (ret < 0) {
                DRM_ERROR("failed to initialize drm fb helper.\n");
                goto err_init;
 
                        "Invalid legacyfb_depth.  Defaulting to 24bpp\n");
                legacyfb_depth = 24;
        }
-       fsl_dev->fbdev = drm_fbdev_cma_init(dev, legacyfb_depth, 1, 1);
+       fsl_dev->fbdev = drm_fbdev_cma_init(dev, legacyfb_depth, 1);
        if (IS_ERR(fsl_dev->fbdev)) {
                ret = PTR_ERR(fsl_dev->fbdev);
                fsl_dev->fbdev = NULL;
 
        drm_fb_helper_prepare(dev, &fbdev->psb_fb_helper, &psb_fb_helper_funcs);
 
        ret = drm_fb_helper_init(dev, &fbdev->psb_fb_helper,
-                                dev_priv->ops->crtcs, INTELFB_CONN_LIMIT);
+                                INTELFB_CONN_LIMIT);
        if (ret)
                goto free;
 
 
                              &hibmc_fbdev_helper_funcs);
 
        /* Now just one crtc and one channel */
-       ret = drm_fb_helper_init(priv->dev,
-                                &hifbdev->helper, 1, 1);
+       ret = drm_fb_helper_init(priv->dev, &hifbdev->helper, 1);
        if (ret) {
                DRM_ERROR("failed to initialize fb helper: %d\n", ret);
                return ret;
 
                drm_fbdev_cma_hotplug_event(priv->fbdev);
        } else {
                priv->fbdev = drm_fbdev_cma_init(dev, 32,
-                               dev->mode_config.num_crtc,
-                               dev->mode_config.num_connector);
+                                                dev->mode_config.num_connector);
                if (IS_ERR(priv->fbdev))
                        priv->fbdev = NULL;
        }
 
        if (!intel_fbdev_init_bios(dev, ifbdev))
                ifbdev->preferred_bpp = 32;
 
-       ret = drm_fb_helper_init(dev, &ifbdev->helper,
-                                INTEL_INFO(dev_priv)->num_pipes, 4);
+       ret = drm_fb_helper_init(dev, &ifbdev->helper, 4);
        if (ret) {
                kfree(ifbdev);
                return ret;
 
                dev_warn(dev, "Invalid legacyfb_depth.  Defaulting to 16bpp\n");
                legacyfb_depth = 16;
        }
-       imxdrm->fbhelper = drm_fbdev_cma_init(drm, legacyfb_depth,
-                               drm->mode_config.num_crtc, MAX_CRTC);
+       imxdrm->fbhelper = drm_fbdev_cma_init(drm, legacyfb_depth, MAX_CRTC);
        if (IS_ERR(imxdrm->fbhelper)) {
                ret = PTR_ERR(imxdrm->fbhelper);
                imxdrm->fbhelper = NULL;
 
        drm->mode_config.funcs = &meson_mode_config_funcs;
 
        priv->fbdev = drm_fbdev_cma_init(drm, 32,
-                                        drm->mode_config.num_crtc,
                                         drm->mode_config.num_connector);
        if (IS_ERR(priv->fbdev)) {
                ret = PTR_ERR(priv->fbdev);
 
        drm_fb_helper_prepare(mdev->dev, &mfbdev->helper, &mga_fb_helper_funcs);
 
        ret = drm_fb_helper_init(mdev->dev, &mfbdev->helper,
-                                mdev->num_crtc, MGAG200FB_CONN_LIMIT);
+                                MGAG200FB_CONN_LIMIT);
        if (ret)
                goto err_fb_helper;
 
 
 
        drm_fb_helper_prepare(dev, helper, &msm_fb_helper_funcs);
 
-       ret = drm_fb_helper_init(dev, helper,
-                       priv->num_crtcs, priv->num_connectors);
+       ret = drm_fb_helper_init(dev, helper, priv->num_connectors);
        if (ret) {
                dev_err(dev->dev, "could not init fbdev: ret=%d\n", ret);
                goto fail;
 
 
        drm_kms_helper_poll_init(drm);
 
-       mxsfb->fbdev = drm_fbdev_cma_init(drm, 32, drm->mode_config.num_crtc,
+       mxsfb->fbdev = drm_fbdev_cma_init(drm, 32,
                                          drm->mode_config.num_connector);
        if (IS_ERR(mxsfb->fbdev)) {
                mxsfb->fbdev = NULL;
 
 
        drm_fb_helper_prepare(dev, &fbcon->helper, &nouveau_fbcon_helper_funcs);
 
-       ret = drm_fb_helper_init(dev, &fbcon->helper,
-                                dev->mode_config.num_crtc, 4);
+       ret = drm_fb_helper_init(dev, &fbcon->helper, 4);
        if (ret)
                goto free;
 
 
 
        drm_fb_helper_prepare(dev, helper, &omap_fb_helper_funcs);
 
-       ret = drm_fb_helper_init(dev, helper,
-                       priv->num_crtcs, priv->num_connectors);
+       ret = drm_fb_helper_init(dev, helper, priv->num_connectors);
        if (ret) {
                dev_err(dev->dev, "could not init fbdev: ret=%d\n", ret);
                goto fail;
 
                              &qxl_fb_helper_funcs);
 
        ret = drm_fb_helper_init(&qdev->ddev, &qfbdev->helper,
-                                qxl_num_crtc,
                                 QXLFB_CONN_LIMIT);
        if (ret)
                goto free;
 
                              &radeon_fb_helper_funcs);
 
        ret = drm_fb_helper_init(rdev->ddev, &rfbdev->helper,
-                                rdev->num_crtc,
                                 RADEONFB_CONN_LIMIT);
        if (ret)
                goto free;
 
        drm_kms_helper_poll_init(dev);
 
        if (dev->mode_config.num_connector) {
-               fbdev = drm_fbdev_cma_init(dev, 32, dev->mode_config.num_crtc,
+               fbdev = drm_fbdev_cma_init(dev, 32,
                                           dev->mode_config.num_connector);
                if (IS_ERR(fbdev))
                        return PTR_ERR(fbdev);
 
 {
        struct rockchip_drm_private *private = dev->dev_private;
        struct drm_fb_helper *helper;
-       unsigned int num_crtc;
        int ret;
 
        if (!dev->mode_config.num_crtc || !dev->mode_config.num_connector)
                return -EINVAL;
 
-       num_crtc = dev->mode_config.num_crtc;
-
        helper = &private->fbdev_helper;
 
        drm_fb_helper_prepare(dev, helper, &rockchip_drm_fb_helper_funcs);
 
-       ret = drm_fb_helper_init(dev, helper, num_crtc, ROCKCHIP_MAX_CONNECTOR);
+       ret = drm_fb_helper_init(dev, helper, ROCKCHIP_MAX_CONNECTOR);
        if (ret < 0) {
                dev_err(dev->dev, "Failed to initialize drm fb helper - %d.\n",
                        ret);
 
 
        private = ddev->dev_private;
        if (ddev->mode_config.num_connector) {
-               fbdev = drm_fbdev_cma_init(ddev, 32, ddev->mode_config.num_crtc,
+               fbdev = drm_fbdev_cma_init(ddev, 32,
                                           ddev->mode_config.num_connector);
                if (IS_ERR(fbdev)) {
                        DRM_DEBUG_DRIVER("Warning: fails to create fbdev\n");
 
 
        drm->mode_config.funcs = &sun4i_de_mode_config_funcs;
 
-       return drm_fbdev_cma_init(drm, 32,
-                                 drm->mode_config.num_crtc,
-                                 drm->mode_config.num_connector);
+       return drm_fbdev_cma_init(drm, 32, drm->mode_config.num_connector);
 }
 
 void sun4i_framebuffer_free(struct drm_device *drm)
 
        struct drm_device *drm = fbdev->base.dev;
        int err;
 
-       err = drm_fb_helper_init(drm, &fbdev->base, num_crtc, max_connectors);
+       err = drm_fb_helper_init(drm, &fbdev->base, max_connectors);
        if (err < 0) {
                dev_err(drm->dev, "failed to initialize DRM FB helper: %d\n",
                        err);
 
        drm_mode_config_reset(ddev);
 
        priv->fbdev = drm_fbdev_cma_init(ddev, bpp,
-                       ddev->mode_config.num_crtc,
-                       ddev->mode_config.num_connector);
+                                        ddev->mode_config.num_connector);
        if (IS_ERR(priv->fbdev)) {
                ret = PTR_ERR(priv->fbdev);
                goto init_failed;
 
 
        drm_fb_helper_prepare(dev, &ufbdev->helper, &udl_fb_helper_funcs);
 
-       ret = drm_fb_helper_init(dev, &ufbdev->helper,
-                                1, 1);
+       ret = drm_fb_helper_init(dev, &ufbdev->helper, 1);
        if (ret)
                goto free;
 
 
        drm_mode_config_reset(dev);
 
        vc4->fbdev = drm_fbdev_cma_init(dev, 32,
-                                       dev->mode_config.num_crtc,
                                        dev->mode_config.num_connector);
        if (IS_ERR(vc4->fbdev))
                vc4->fbdev = NULL;
 
        drm_fb_helper_prepare(vgdev->ddev, &vgfbdev->helper,
                              &virtio_gpu_fb_helper_funcs);
        ret = drm_fb_helper_init(vgdev->ddev, &vgfbdev->helper,
-                                vgdev->num_scanouts,
                                 VIRTIO_GPUFB_CONN_LIMIT);
        if (ret) {
                kfree(vgfbdev);
 
        drm_mode_config_reset(drm);
        drm_kms_helper_poll_init(drm);
 
-       priv->fbdev = drm_fbdev_cma_init(drm, 32, drm->mode_config.num_crtc,
+       priv->fbdev = drm_fbdev_cma_init(drm, 32,
                                         drm->mode_config.num_connector);
        if (IS_ERR(priv->fbdev)) {
                ret = PTR_ERR(priv->fbdev);
 
 struct drm_plane_state;
 
 struct drm_fbdev_cma *drm_fbdev_cma_init_with_funcs(struct drm_device *dev,
-       unsigned int preferred_bpp, unsigned int num_crtc,
-       unsigned int max_conn_count, const struct drm_framebuffer_funcs *funcs);
+       unsigned int preferred_bpp, unsigned int max_conn_count,
+       const struct drm_framebuffer_funcs *funcs);
 struct drm_fbdev_cma *drm_fbdev_cma_init(struct drm_device *dev,
-       unsigned int preferred_bpp, unsigned int num_crtc,
-       unsigned int max_conn_count);
+       unsigned int preferred_bpp, unsigned int max_conn_count);
 void drm_fbdev_cma_fini(struct drm_fbdev_cma *fbdev_cma);
 
 void drm_fbdev_cma_restore_mode(struct drm_fbdev_cma *fbdev_cma);
 
 void drm_fb_helper_prepare(struct drm_device *dev, struct drm_fb_helper *helper,
                           const struct drm_fb_helper_funcs *funcs);
 int drm_fb_helper_init(struct drm_device *dev,
-                      struct drm_fb_helper *helper, int crtc_count,
-                      int max_conn);
+                      struct drm_fb_helper *helper, int max_conn);
 void drm_fb_helper_fini(struct drm_fb_helper *helper);
 int drm_fb_helper_blank(int blank, struct fb_info *info);
 int drm_fb_helper_pan_display(struct fb_var_screeninfo *var,