diff options
Diffstat (limited to 'drivers/staging/imx-drm')
23 files changed, 4011 insertions, 4786 deletions
diff --git a/drivers/staging/imx-drm/Kconfig b/drivers/staging/imx-drm/Kconfig index 394254f7d6b..82fb758a29b 100644 --- a/drivers/staging/imx-drm/Kconfig +++ b/drivers/staging/imx-drm/Kconfig @@ -1,6 +1,7 @@  config DRM_IMX  	tristate "DRM Support for Freescale i.MX"  	select DRM_KMS_HELPER +	select DRM_KMS_FB_HELPER  	select VIDEOMODE_HELPERS  	select DRM_GEM_CMA_HELPER  	select DRM_KMS_CMA_HELPER @@ -19,6 +20,7 @@ config DRM_IMX_FB_HELPER  config DRM_IMX_PARALLEL_DISPLAY  	tristate "Support for parallel displays" +	select DRM_PANEL  	depends on DRM_IMX  	select VIDEOMODE_HELPERS @@ -37,18 +39,15 @@ config DRM_IMX_LDB  	  Choose this to enable the internal LVDS Display Bridge (LDB)  	  found on i.MX53 and i.MX6 processors. -config DRM_IMX_IPUV3_CORE -	tristate "IPUv3 core support" -	depends on DRM_IMX -	depends on RESET_CONTROLLER -	help -	  Choose this if you have a i.MX5/6 system and want -	  to use the IPU. This option only enables IPU base -	  support. -  config DRM_IMX_IPUV3  	tristate "DRM Support for i.MX IPUv3"  	depends on DRM_IMX -	depends on DRM_IMX_IPUV3_CORE +	depends on IMX_IPUV3_CORE  	help  	  Choose this if you have a i.MX5 or i.MX6 processor. + +config DRM_IMX_HDMI +	tristate "Freescale i.MX DRM HDMI" +	depends on DRM_IMX +	help +	  Choose this if you want to use HDMI on i.MX6. diff --git a/drivers/staging/imx-drm/Makefile b/drivers/staging/imx-drm/Makefile index bfaf69378ac..582c438d8cb 100644 --- a/drivers/staging/imx-drm/Makefile +++ b/drivers/staging/imx-drm/Makefile @@ -1,11 +1,12 @@ -imxdrm-objs := imx-drm-core.o imx-fb.o +imxdrm-objs := imx-drm-core.o  obj-$(CONFIG_DRM_IMX) += imxdrm.o  obj-$(CONFIG_DRM_IMX_PARALLEL_DISPLAY) += parallel-display.o  obj-$(CONFIG_DRM_IMX_TVE) += imx-tve.o  obj-$(CONFIG_DRM_IMX_LDB) += imx-ldb.o -obj-$(CONFIG_DRM_IMX_FB_HELPER) += imx-fbdev.o -obj-$(CONFIG_DRM_IMX_IPUV3_CORE) += ipu-v3/ -obj-$(CONFIG_DRM_IMX_IPUV3)	+= ipuv3-crtc.o + +imx-ipuv3-crtc-objs  := ipuv3-crtc.o ipuv3-plane.o +obj-$(CONFIG_DRM_IMX_IPUV3)	+= imx-ipuv3-crtc.o +obj-$(CONFIG_DRM_IMX_HDMI) += imx-hdmi.o diff --git a/drivers/staging/imx-drm/TODO b/drivers/staging/imx-drm/TODO index 9cfa2a7efdc..29636fb1395 100644 --- a/drivers/staging/imx-drm/TODO +++ b/drivers/staging/imx-drm/TODO @@ -1,16 +1,10 @@  TODO:  - get DRM Maintainer review for this code -- Wait for common display framework to hit mainline and update the IPU -  driver to use it. This will most probably make changes to the devicetree -  bindings necessary. -- Factor out more code to common helper functions  - decide where to put the base driver. It is not specific to a subsystem    and would be used by DRM/KMS and media/V4L2  Missing features (not necessarily for moving out of staging): -- Add KMS plane support for CRTC driver -- Add i.MX6 HDMI support  - Add support for IC (Image converter)  - Add support for CSI (CMOS Sensor interface)  - Add support for VDIC (Video Deinterlacer) diff --git a/drivers/staging/imx-drm/imx-drm-core.c b/drivers/staging/imx-drm/imx-drm-core.c index 47c5888461f..def8280d7ee 100644 --- a/drivers/staging/imx-drm/imx-drm-core.c +++ b/drivers/staging/imx-drm/imx-drm-core.c @@ -13,14 +13,15 @@   * GNU General Public License for more details.   *   */ - +#include <linux/component.h>  #include <linux/device.h> +#include <linux/fb.h> +#include <linux/module.h> +#include <linux/of_graph.h>  #include <linux/platform_device.h>  #include <drm/drmP.h>  #include <drm/drm_fb_helper.h>  #include <drm/drm_crtc_helper.h> -#include <linux/fb.h> -#include <linux/module.h>  #include <drm/drm_gem_cma_helper.h>  #include <drm/drm_fb_cma_helper.h> @@ -28,143 +29,128 @@  #define MAX_CRTC	4 -struct crtc_cookie { -	void *cookie; -	int id; +struct imx_drm_crtc; + +struct imx_drm_component { +	struct device_node *of_node;  	struct list_head list;  };  struct imx_drm_device {  	struct drm_device			*drm; -	struct device				*dev; -	struct list_head			crtc_list; -	struct list_head			encoder_list; -	struct list_head			connector_list; -	struct mutex				mutex; -	int					references; +	struct imx_drm_crtc			*crtc[MAX_CRTC];  	int					pipes;  	struct drm_fbdev_cma			*fbhelper;  };  struct imx_drm_crtc {  	struct drm_crtc				*crtc; -	struct list_head			list; -	struct imx_drm_device			*imxdrm;  	int					pipe;  	struct imx_drm_crtc_helper_funcs	imx_drm_helper_funcs; -	struct module				*owner; -	struct crtc_cookie			cookie; +	struct device_node			*port;  }; -struct imx_drm_encoder { -	struct drm_encoder			*encoder; -	struct list_head			list; -	struct module				*owner; -	struct list_head			possible_crtcs; -}; +static int legacyfb_depth = 16; +module_param(legacyfb_depth, int, 0444); -struct imx_drm_connector { -	struct drm_connector			*connector; -	struct list_head			list; -	struct module				*owner; -}; +int imx_drm_crtc_id(struct imx_drm_crtc *crtc) +{ +	return crtc->pipe; +} +EXPORT_SYMBOL_GPL(imx_drm_crtc_id);  static void imx_drm_driver_lastclose(struct drm_device *drm)  { +#if IS_ENABLED(CONFIG_DRM_IMX_FB_HELPER)  	struct imx_drm_device *imxdrm = drm->dev_private;  	if (imxdrm->fbhelper)  		drm_fbdev_cma_restore_mode(imxdrm->fbhelper); +#endif  }  static int imx_drm_driver_unload(struct drm_device *drm)  { +#if IS_ENABLED(CONFIG_DRM_IMX_FB_HELPER)  	struct imx_drm_device *imxdrm = drm->dev_private; +#endif -	imx_drm_device_put(); +	drm_kms_helper_poll_fini(drm); -	drm_mode_config_cleanup(imxdrm->drm); -	drm_kms_helper_poll_fini(imxdrm->drm); +#if IS_ENABLED(CONFIG_DRM_IMX_FB_HELPER) +	if (imxdrm->fbhelper) +		drm_fbdev_cma_fini(imxdrm->fbhelper); +#endif + +	component_unbind_all(drm->dev, drm); + +	drm_vblank_cleanup(drm); +	drm_mode_config_cleanup(drm);  	return 0;  } -/* - * We don't care at all for crtc numbers, but the core expects the - * crtcs to be numbered - */ -static struct imx_drm_crtc *imx_drm_crtc_by_num(struct imx_drm_device *imxdrm, -		int num) +static struct imx_drm_crtc *imx_drm_find_crtc(struct drm_crtc *crtc)  { -	struct imx_drm_crtc *imx_drm_crtc; +	struct imx_drm_device *imxdrm = crtc->dev->dev_private; +	unsigned i; + +	for (i = 0; i < MAX_CRTC; i++) +		if (imxdrm->crtc[i] && imxdrm->crtc[i]->crtc == crtc) +			return imxdrm->crtc[i]; -	list_for_each_entry(imx_drm_crtc, &imxdrm->crtc_list, list) -		if (imx_drm_crtc->pipe == num) -			return imx_drm_crtc;  	return NULL;  } -int imx_drm_crtc_panel_format_pins(struct drm_crtc *crtc, u32 encoder_type, +int imx_drm_panel_format_pins(struct drm_encoder *encoder,  		u32 interface_pix_fmt, int hsync_pin, int vsync_pin)  { -	struct imx_drm_device *imxdrm = crtc->dev->dev_private; -	struct imx_drm_crtc *imx_crtc;  	struct imx_drm_crtc_helper_funcs *helper; +	struct imx_drm_crtc *imx_crtc; -	mutex_lock(&imxdrm->mutex); - -	list_for_each_entry(imx_crtc, &imxdrm->crtc_list, list) -		if (imx_crtc->crtc == crtc) -			goto found; - -	mutex_unlock(&imxdrm->mutex); - -	return -EINVAL; -found: -	mutex_unlock(&imxdrm->mutex); +	imx_crtc = imx_drm_find_crtc(encoder->crtc); +	if (!imx_crtc) +		return -EINVAL;  	helper = &imx_crtc->imx_drm_helper_funcs;  	if (helper->set_interface_pix_fmt) -		return helper->set_interface_pix_fmt(crtc, -				encoder_type, interface_pix_fmt, +		return helper->set_interface_pix_fmt(encoder->crtc, +				encoder->encoder_type, interface_pix_fmt,  				hsync_pin, vsync_pin);  	return 0;  } -EXPORT_SYMBOL_GPL(imx_drm_crtc_panel_format_pins); +EXPORT_SYMBOL_GPL(imx_drm_panel_format_pins); -int imx_drm_crtc_panel_format(struct drm_crtc *crtc, u32 encoder_type, -		u32 interface_pix_fmt) +int imx_drm_panel_format(struct drm_encoder *encoder, u32 interface_pix_fmt)  { -	return imx_drm_crtc_panel_format_pins(crtc, encoder_type, -					      interface_pix_fmt, 2, 3); +	return imx_drm_panel_format_pins(encoder, interface_pix_fmt, 2, 3);  } -EXPORT_SYMBOL_GPL(imx_drm_crtc_panel_format); +EXPORT_SYMBOL_GPL(imx_drm_panel_format);  int imx_drm_crtc_vblank_get(struct imx_drm_crtc *imx_drm_crtc)  { -	return drm_vblank_get(imx_drm_crtc->imxdrm->drm, imx_drm_crtc->pipe); +	return drm_vblank_get(imx_drm_crtc->crtc->dev, imx_drm_crtc->pipe);  }  EXPORT_SYMBOL_GPL(imx_drm_crtc_vblank_get);  void imx_drm_crtc_vblank_put(struct imx_drm_crtc *imx_drm_crtc)  { -	drm_vblank_put(imx_drm_crtc->imxdrm->drm, imx_drm_crtc->pipe); +	drm_vblank_put(imx_drm_crtc->crtc->dev, imx_drm_crtc->pipe);  }  EXPORT_SYMBOL_GPL(imx_drm_crtc_vblank_put);  void imx_drm_handle_vblank(struct imx_drm_crtc *imx_drm_crtc)  { -	drm_handle_vblank(imx_drm_crtc->imxdrm->drm, imx_drm_crtc->pipe); +	drm_handle_vblank(imx_drm_crtc->crtc->dev, imx_drm_crtc->pipe);  }  EXPORT_SYMBOL_GPL(imx_drm_handle_vblank);  static int imx_drm_enable_vblank(struct drm_device *drm, int crtc)  {  	struct imx_drm_device *imxdrm = drm->dev_private; -	struct imx_drm_crtc *imx_drm_crtc; +	struct imx_drm_crtc *imx_drm_crtc = imxdrm->crtc[crtc];  	int ret; -	imx_drm_crtc = imx_drm_crtc_by_num(imxdrm, crtc);  	if (!imx_drm_crtc)  		return -EINVAL; @@ -180,9 +166,8 @@ static int imx_drm_enable_vblank(struct drm_device *drm, int crtc)  static void imx_drm_disable_vblank(struct drm_device *drm, int crtc)  {  	struct imx_drm_device *imxdrm = drm->dev_private; -	struct imx_drm_crtc *imx_drm_crtc; +	struct imx_drm_crtc *imx_drm_crtc = imxdrm->crtc[crtc]; -	imx_drm_crtc = imx_drm_crtc_by_num(imxdrm, crtc);  	if (!imx_drm_crtc)  		return; @@ -192,6 +177,18 @@ static void imx_drm_disable_vblank(struct drm_device *drm, int crtc)  	imx_drm_crtc->imx_drm_helper_funcs.disable_vblank(imx_drm_crtc->crtc);  } +static void imx_drm_driver_preclose(struct drm_device *drm, +		struct drm_file *file) +{ +	int i; + +	if (!file->is_master) +		return; + +	for (i = 0; i < MAX_CRTC; i++) +		imx_drm_disable_vblank(drm, i); +} +  static const struct file_operations imx_drm_driver_fops = {  	.owner = THIS_MODULE,  	.open = drm_open, @@ -203,328 +200,192 @@ static const struct file_operations imx_drm_driver_fops = {  	.llseek = noop_llseek,  }; -static struct imx_drm_device *imx_drm_device; - -static struct imx_drm_device *__imx_drm_device(void) -{ -	return imx_drm_device; -} - -struct drm_device *imx_drm_device_get(void) -{ -	struct imx_drm_device *imxdrm = __imx_drm_device(); -	struct imx_drm_encoder *enc; -	struct imx_drm_connector *con; -	struct imx_drm_crtc *crtc; - -	list_for_each_entry(enc, &imxdrm->encoder_list, list) { -		if (!try_module_get(enc->owner)) { -			dev_err(imxdrm->dev, "could not get module %s\n", -					module_name(enc->owner)); -			goto unwind_enc; -		} -	} - -	list_for_each_entry(con, &imxdrm->connector_list, list) { -		if (!try_module_get(con->owner)) { -			dev_err(imxdrm->dev, "could not get module %s\n", -					module_name(con->owner)); -			goto unwind_con; -		} -	} - -	list_for_each_entry(crtc, &imxdrm->crtc_list, list) { -		if (!try_module_get(crtc->owner)) { -			dev_err(imxdrm->dev, "could not get module %s\n", -					module_name(crtc->owner)); -			goto unwind_crtc; -		} -	} - -	imxdrm->references++; - -	return imxdrm->drm; - -unwind_crtc: -	list_for_each_entry_continue_reverse(crtc, &imxdrm->crtc_list, list) -		module_put(crtc->owner); -unwind_con: -	list_for_each_entry_continue_reverse(con, &imxdrm->connector_list, list) -		module_put(con->owner); -unwind_enc: -	list_for_each_entry_continue_reverse(enc, &imxdrm->encoder_list, list) -		module_put(enc->owner); - -	mutex_unlock(&imxdrm->mutex); - -	return NULL; - -} -EXPORT_SYMBOL_GPL(imx_drm_device_get); - -void imx_drm_device_put(void) -{ -	struct imx_drm_device *imxdrm = __imx_drm_device(); -	struct imx_drm_encoder *enc; -	struct imx_drm_connector *con; -	struct imx_drm_crtc *crtc; - -	mutex_lock(&imxdrm->mutex); - -	list_for_each_entry(crtc, &imxdrm->crtc_list, list) -		module_put(crtc->owner); - -	list_for_each_entry(con, &imxdrm->connector_list, list) -		module_put(con->owner); - -	list_for_each_entry(enc, &imxdrm->encoder_list, list) -		module_put(enc->owner); - -	imxdrm->references--; - -	mutex_unlock(&imxdrm->mutex); -} -EXPORT_SYMBOL_GPL(imx_drm_device_put); - -static int drm_mode_group_reinit(struct drm_device *dev) -{ -	struct drm_mode_group *group = &dev->primary->mode_group; -	uint32_t *id_list = group->id_list; -	int ret; - -	ret = drm_mode_group_init_legacy_group(dev, group); -	if (ret < 0) -		return ret; - -	kfree(id_list); -	return 0; -} - -/* - * register an encoder to the drm core - */ -static int imx_drm_encoder_register(struct imx_drm_encoder *imx_drm_encoder) -{ -	struct imx_drm_device *imxdrm = __imx_drm_device(); - -	INIT_LIST_HEAD(&imx_drm_encoder->possible_crtcs); - -	drm_encoder_init(imxdrm->drm, imx_drm_encoder->encoder, -			imx_drm_encoder->encoder->funcs, -			imx_drm_encoder->encoder->encoder_type); - -	drm_mode_group_reinit(imxdrm->drm); - -	return 0; -} - -/* - * unregister an encoder from the drm core - */ -static void imx_drm_encoder_unregister(struct imx_drm_encoder -		*imx_drm_encoder) +void imx_drm_connector_destroy(struct drm_connector *connector)  { -	struct imx_drm_device *imxdrm = __imx_drm_device(); - -	drm_encoder_cleanup(imx_drm_encoder->encoder); - -	drm_mode_group_reinit(imxdrm->drm); +	drm_sysfs_connector_remove(connector); +	drm_connector_cleanup(connector);  } +EXPORT_SYMBOL_GPL(imx_drm_connector_destroy); -/* - * register a connector to the drm core - */ -static int imx_drm_connector_register( -		struct imx_drm_connector *imx_drm_connector) +void imx_drm_encoder_destroy(struct drm_encoder *encoder)  { -	struct imx_drm_device *imxdrm = __imx_drm_device(); - -	drm_connector_init(imxdrm->drm, imx_drm_connector->connector, -			imx_drm_connector->connector->funcs, -			imx_drm_connector->connector->connector_type); -	drm_mode_group_reinit(imxdrm->drm); - -	return drm_sysfs_connector_add(imx_drm_connector->connector); +	drm_encoder_cleanup(encoder);  } +EXPORT_SYMBOL_GPL(imx_drm_encoder_destroy); -/* - * unregister a connector from the drm core - */ -static void imx_drm_connector_unregister( -		struct imx_drm_connector *imx_drm_connector) +static void imx_drm_output_poll_changed(struct drm_device *drm)  { -	struct imx_drm_device *imxdrm = __imx_drm_device(); - -	drm_sysfs_connector_remove(imx_drm_connector->connector); -	drm_connector_cleanup(imx_drm_connector->connector); +#if IS_ENABLED(CONFIG_DRM_IMX_FB_HELPER) +	struct imx_drm_device *imxdrm = drm->dev_private; -	drm_mode_group_reinit(imxdrm->drm); +	drm_fbdev_cma_hotplug_event(imxdrm->fbhelper); +#endif  } -/* - * register a crtc to the drm core - */ -static int imx_drm_crtc_register(struct imx_drm_crtc *imx_drm_crtc) -{ -	struct imx_drm_device *imxdrm = __imx_drm_device(); -	int ret; - -	drm_crtc_init(imxdrm->drm, imx_drm_crtc->crtc, -			imx_drm_crtc->imx_drm_helper_funcs.crtc_funcs); -	ret = drm_mode_crtc_set_gamma_size(imx_drm_crtc->crtc, 256); -	if (ret) -		return ret; - -	drm_crtc_helper_add(imx_drm_crtc->crtc, -			imx_drm_crtc->imx_drm_helper_funcs.crtc_helper_funcs); - -	drm_mode_group_reinit(imxdrm->drm); - -	return 0; -} +static struct drm_mode_config_funcs imx_drm_mode_config_funcs = { +	.fb_create = drm_fb_cma_create, +	.output_poll_changed = imx_drm_output_poll_changed, +};  /* - * Called by the CRTC driver when all CRTCs are registered. This - * puts all the pieces together and initializes the driver. - * Once this is called no more CRTCs can be registered since - * the drm core has hardcoded the number of crtcs in several - * places. + * Main DRM initialisation. This binds, initialises and registers + * with DRM the subcomponents of the driver.   */  static int imx_drm_driver_load(struct drm_device *drm, unsigned long flags)  { -	struct imx_drm_device *imxdrm = __imx_drm_device(); +	struct imx_drm_device *imxdrm; +	struct drm_connector *connector;  	int ret; +	imxdrm = devm_kzalloc(drm->dev, sizeof(*imxdrm), GFP_KERNEL); +	if (!imxdrm) +		return -ENOMEM; +  	imxdrm->drm = drm;  	drm->dev_private = imxdrm;  	/*  	 * enable drm irq mode. -	 * - with irq_enabled = 1, we can use the vblank feature. +	 * - with irq_enabled = true, we can use the vblank feature.  	 *  	 * P.S. note that we wouldn't use drm irq handler but  	 *      just specific driver own one instead because  	 *      drm framework supports only one irq handler and  	 *      drivers can well take care of their interrupts  	 */ -	drm->irq_enabled = 1; +	drm->irq_enabled = true; + +	/* +	 * set max width and height as default value(4096x4096). +	 * this value would be used to check framebuffer size limitation +	 * at drm_mode_addfb(). +	 */ +	drm->mode_config.min_width = 64; +	drm->mode_config.min_height = 64; +	drm->mode_config.max_width = 4096; +	drm->mode_config.max_height = 4096; +	drm->mode_config.funcs = &imx_drm_mode_config_funcs;  	drm_mode_config_init(drm); -	imx_drm_mode_config_init(drm); -	mutex_lock(&imxdrm->mutex); +	ret = drm_vblank_init(drm, MAX_CRTC); +	if (ret) +		goto err_kms; -	drm_kms_helper_poll_init(imxdrm->drm); +	/* +	 * with vblank_disable_allowed = true, vblank interrupt will be +	 * disabled by drm timer once a current process gives up ownership +	 * of vblank event. (after drm_vblank_put function is called) +	 */ +	drm->vblank_disable_allowed = true; -	/* setup the grouping for the legacy output */ -	ret = drm_mode_group_init_legacy_group(imxdrm->drm, -			&imxdrm->drm->primary->mode_group); -	if (ret) -		goto err_init; +	platform_set_drvdata(drm->platformdev, drm); -	ret = drm_vblank_init(imxdrm->drm, MAX_CRTC); +	/* Now try and bind all our sub-components */ +	ret = component_bind_all(drm->dev, drm);  	if (ret) -		goto err_init; +		goto err_vblank;  	/* -	 * with vblank_disable_allowed = 1, vblank interrupt will be disabled -	 * by drm timer once a current process gives up ownership of -	 * vblank event.(after drm_vblank_put function is called) +	 * All components are now added, we can publish the connector sysfs +	 * entries to userspace.  This will generate hotplug events and so +	 * userspace will expect to be able to access DRM at this point.  	 */ -	imxdrm->drm->vblank_disable_allowed = 1; +	list_for_each_entry(connector, &drm->mode_config.connector_list, head) { +		ret = drm_sysfs_connector_add(connector); +		if (ret) { +			dev_err(drm->dev, +				"[CONNECTOR:%d:%s] drm_sysfs_connector_add failed: %d\n", +				connector->base.id, +				connector->name, ret); +			goto err_unbind; +		} +	} -	if (!imx_drm_device_get()) -		ret = -EINVAL; +	/* +	 * All components are now initialised, so setup the fb helper. +	 * The fb helper takes copies of key hardware information, so the +	 * crtcs/connectors/encoders must not change after this point. +	 */ +#if IS_ENABLED(CONFIG_DRM_IMX_FB_HELPER) +	if (legacyfb_depth != 16 && legacyfb_depth != 32) { +		dev_warn(drm->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); +	if (IS_ERR(imxdrm->fbhelper)) { +		ret = PTR_ERR(imxdrm->fbhelper); +		imxdrm->fbhelper = NULL; +		goto err_unbind; +	} +#endif -	ret = 0; +	drm_kms_helper_poll_init(drm); -err_init: -	mutex_unlock(&imxdrm->mutex); +	return 0; -	return ret; -} +err_unbind: +	component_unbind_all(drm->dev, drm); +err_vblank: +	drm_vblank_cleanup(drm); +err_kms: +	drm_mode_config_cleanup(drm); -static void imx_drm_update_possible_crtcs(void) -{ -	struct imx_drm_device *imxdrm = __imx_drm_device(); -	struct imx_drm_crtc *imx_drm_crtc; -	struct imx_drm_encoder *enc; -	struct crtc_cookie *cookie; - -	list_for_each_entry(enc, &imxdrm->encoder_list, list) { -		u32 possible_crtcs = 0; - -		list_for_each_entry(cookie, &enc->possible_crtcs, list) { -			list_for_each_entry(imx_drm_crtc, &imxdrm->crtc_list, list) { -				if (imx_drm_crtc->cookie.cookie == cookie->cookie && -						imx_drm_crtc->cookie.id == cookie->id) { -					possible_crtcs |= 1 << imx_drm_crtc->pipe; -				} -			} -		} -		enc->encoder->possible_crtcs = possible_crtcs; -		enc->encoder->possible_clones = possible_crtcs; -	} +	return ret;  }  /*   * imx_drm_add_crtc - add a new crtc - * - * The return value if !NULL is a cookie for the caller to pass to - * imx_drm_remove_crtc later.   */ -int imx_drm_add_crtc(struct drm_crtc *crtc, +int imx_drm_add_crtc(struct drm_device *drm, struct drm_crtc *crtc,  		struct imx_drm_crtc **new_crtc,  		const struct imx_drm_crtc_helper_funcs *imx_drm_helper_funcs, -		struct module *owner, void *cookie, int id) +		struct device_node *port)  { -	struct imx_drm_device *imxdrm = __imx_drm_device(); +	struct imx_drm_device *imxdrm = drm->dev_private;  	struct imx_drm_crtc *imx_drm_crtc;  	int ret; -	mutex_lock(&imxdrm->mutex); +	/* +	 * The vblank arrays are dimensioned by MAX_CRTC - we can't +	 * pass IDs greater than this to those functions. +	 */ +	if (imxdrm->pipes >= MAX_CRTC) +		return -EINVAL; -	if (imxdrm->references) { -		ret = -EBUSY; -		goto err_busy; -	} +	if (imxdrm->drm->open_count) +		return -EBUSY;  	imx_drm_crtc = kzalloc(sizeof(*imx_drm_crtc), GFP_KERNEL); -	if (!imx_drm_crtc) { -		ret = -ENOMEM; -		goto err_alloc; -	} +	if (!imx_drm_crtc) +		return -ENOMEM;  	imx_drm_crtc->imx_drm_helper_funcs = *imx_drm_helper_funcs;  	imx_drm_crtc->pipe = imxdrm->pipes++; -	imx_drm_crtc->cookie.cookie = cookie; -	imx_drm_crtc->cookie.id = id; - +	imx_drm_crtc->port = port;  	imx_drm_crtc->crtc = crtc; -	imx_drm_crtc->imxdrm = imxdrm; - -	imx_drm_crtc->owner = owner; -	list_add_tail(&imx_drm_crtc->list, &imxdrm->crtc_list); +	imxdrm->crtc[imx_drm_crtc->pipe] = imx_drm_crtc;  	*new_crtc = imx_drm_crtc; -	ret = imx_drm_crtc_register(imx_drm_crtc); +	ret = drm_mode_crtc_set_gamma_size(imx_drm_crtc->crtc, 256);  	if (ret)  		goto err_register; -	imx_drm_update_possible_crtcs(); +	drm_crtc_helper_add(crtc, +			imx_drm_crtc->imx_drm_helper_funcs.crtc_helper_funcs); -	mutex_unlock(&imxdrm->mutex); +	drm_crtc_init(drm, crtc, +			imx_drm_crtc->imx_drm_helper_funcs.crtc_funcs);  	return 0;  err_register: +	imxdrm->crtc[imx_drm_crtc->pipe] = NULL;  	kfree(imx_drm_crtc); -err_alloc: -err_busy: -	mutex_unlock(&imxdrm->mutex);  	return ret;  }  EXPORT_SYMBOL_GPL(imx_drm_add_crtc); @@ -534,17 +395,11 @@ EXPORT_SYMBOL_GPL(imx_drm_add_crtc);   */  int imx_drm_remove_crtc(struct imx_drm_crtc *imx_drm_crtc)  { -	struct imx_drm_device *imxdrm = imx_drm_crtc->imxdrm; - -	mutex_lock(&imxdrm->mutex); +	struct imx_drm_device *imxdrm = imx_drm_crtc->crtc->dev->dev_private;  	drm_crtc_cleanup(imx_drm_crtc->crtc); -	list_del(&imx_drm_crtc->list); - -	drm_mode_group_reinit(imxdrm->drm); - -	mutex_unlock(&imxdrm->mutex); +	imxdrm->crtc[imx_drm_crtc->pipe] = NULL;  	kfree(imx_drm_crtc); @@ -553,242 +408,141 @@ int imx_drm_remove_crtc(struct imx_drm_crtc *imx_drm_crtc)  EXPORT_SYMBOL_GPL(imx_drm_remove_crtc);  /* - * imx_drm_add_encoder - add a new encoder + * Find the DRM CRTC possible mask for the connected endpoint. + * + * The encoder possible masks are defined by their position in the + * mode_config crtc_list.  This means that CRTCs must not be added + * or removed once the DRM device has been fully initialised.   */ -int imx_drm_add_encoder(struct drm_encoder *encoder, -		struct imx_drm_encoder **newenc, struct module *owner) +static uint32_t imx_drm_find_crtc_mask(struct imx_drm_device *imxdrm, +	struct device_node *endpoint)  { -	struct imx_drm_device *imxdrm = __imx_drm_device(); -	struct imx_drm_encoder *imx_drm_encoder; -	int ret; - -	mutex_lock(&imxdrm->mutex); +	struct device_node *port; +	unsigned i; -	if (imxdrm->references) { -		ret = -EBUSY; -		goto err_busy; -	} +	port = of_graph_get_remote_port(endpoint); +	if (!port) +		return 0; +	of_node_put(port); -	imx_drm_encoder = kzalloc(sizeof(*imx_drm_encoder), GFP_KERNEL); -	if (!imx_drm_encoder) { -		ret = -ENOMEM; -		goto err_alloc; +	for (i = 0; i < MAX_CRTC; i++) { +		struct imx_drm_crtc *imx_drm_crtc = imxdrm->crtc[i]; +		if (imx_drm_crtc && imx_drm_crtc->port == port) +			return drm_crtc_mask(imx_drm_crtc->crtc);  	} -	imx_drm_encoder->encoder = encoder; -	imx_drm_encoder->owner = owner; - -	ret = imx_drm_encoder_register(imx_drm_encoder); -	if (ret) { -		ret = -ENOMEM; -		goto err_register; -	} - -	list_add_tail(&imx_drm_encoder->list, &imxdrm->encoder_list); - -	*newenc = imx_drm_encoder; - -	mutex_unlock(&imxdrm->mutex); -  	return 0; +} -err_register: -	kfree(imx_drm_encoder); -err_alloc: -err_busy: -	mutex_unlock(&imxdrm->mutex); - -	return ret; +static struct device_node *imx_drm_of_get_next_endpoint( +		const struct device_node *parent, struct device_node *prev) +{ +	struct device_node *node = of_graph_get_next_endpoint(parent, prev); +	of_node_put(prev); +	return node;  } -EXPORT_SYMBOL_GPL(imx_drm_add_encoder); -int imx_drm_encoder_add_possible_crtcs( -		struct imx_drm_encoder *imx_drm_encoder, -		struct device_node *np) +int imx_drm_encoder_parse_of(struct drm_device *drm, +	struct drm_encoder *encoder, struct device_node *np)  { -	struct imx_drm_device *imxdrm = __imx_drm_device(); -	struct of_phandle_args args; -	struct crtc_cookie *c; -	int ret = 0; +	struct imx_drm_device *imxdrm = drm->dev_private; +	struct device_node *ep = NULL; +	uint32_t crtc_mask = 0;  	int i; -	if (!list_empty(&imx_drm_encoder->possible_crtcs)) -		return -EBUSY; +	for (i = 0; ; i++) { +		u32 mask; -	for (i = 0; !ret; i++) { -		ret = of_parse_phandle_with_args(np, "crtcs", -				"#crtc-cells", i, &args); -		if (ret < 0) +		ep = imx_drm_of_get_next_endpoint(np, ep); +		if (!ep)  			break; -		c = kzalloc(sizeof(*c), GFP_KERNEL); -		if (!c) { -			of_node_put(args.np); -			return -ENOMEM; -		} - -		c->cookie = args.np; -		c->id = args.args_count > 0 ? args.args[0] : 0; - -		of_node_put(args.np); - -		mutex_lock(&imxdrm->mutex); +		mask = imx_drm_find_crtc_mask(imxdrm, ep); -		list_add_tail(&c->list, &imx_drm_encoder->possible_crtcs); +		/* +		 * If we failed to find the CRTC(s) which this encoder is +		 * supposed to be connected to, it's because the CRTC has +		 * not been registered yet.  Defer probing, and hope that +		 * the required CRTC is added later. +		 */ +		if (mask == 0) +			return -EPROBE_DEFER; -		mutex_unlock(&imxdrm->mutex); +		crtc_mask |= mask;  	} -	imx_drm_update_possible_crtcs(); +	if (ep) +		of_node_put(ep); +	if (i == 0) +		return -ENOENT; -	return 0; -} -EXPORT_SYMBOL_GPL(imx_drm_encoder_add_possible_crtcs); +	encoder->possible_crtcs = crtc_mask; -int imx_drm_encoder_get_mux_id(struct imx_drm_encoder *imx_drm_encoder, -		struct drm_crtc *crtc) -{ -	struct imx_drm_device *imxdrm = __imx_drm_device(); -	struct imx_drm_crtc *imx_crtc; -	int i = 0; - -	mutex_lock(&imxdrm->mutex); - -	list_for_each_entry(imx_crtc, &imxdrm->crtc_list, list) { -		if (imx_crtc->crtc == crtc) -			goto found; -		i++; -	} - -	mutex_unlock(&imxdrm->mutex); - -	return -EINVAL; -found: -	mutex_unlock(&imxdrm->mutex); - -	return i; -} -EXPORT_SYMBOL_GPL(imx_drm_encoder_get_mux_id); - -/* - * imx_drm_remove_encoder - remove an encoder - */ -int imx_drm_remove_encoder(struct imx_drm_encoder *imx_drm_encoder) -{ -	struct imx_drm_device *imxdrm = __imx_drm_device(); -	struct crtc_cookie *c, *tmp; - -	mutex_lock(&imxdrm->mutex); - -	imx_drm_encoder_unregister(imx_drm_encoder); - -	list_del(&imx_drm_encoder->list); - -	list_for_each_entry_safe(c, tmp, &imx_drm_encoder->possible_crtcs, -			list) -		kfree(c); - -	mutex_unlock(&imxdrm->mutex); - -	kfree(imx_drm_encoder); +	/* FIXME: this is the mask of outputs which can clone this output. */ +	encoder->possible_clones = ~0;  	return 0;  } -EXPORT_SYMBOL_GPL(imx_drm_remove_encoder); +EXPORT_SYMBOL_GPL(imx_drm_encoder_parse_of);  /* - * imx_drm_add_connector - add a connector + * @node: device tree node containing encoder input ports + * @encoder: drm_encoder   */ -int imx_drm_add_connector(struct drm_connector *connector, -		struct imx_drm_connector **new_con, -		struct module *owner) +int imx_drm_encoder_get_mux_id(struct device_node *node, +			       struct drm_encoder *encoder)  { -	struct imx_drm_device *imxdrm = __imx_drm_device(); -	struct imx_drm_connector *imx_drm_connector; +	struct imx_drm_crtc *imx_crtc = imx_drm_find_crtc(encoder->crtc); +	struct device_node *ep = NULL; +	struct of_endpoint endpoint; +	struct device_node *port;  	int ret; -	mutex_lock(&imxdrm->mutex); - -	if (imxdrm->references) { -		ret = -EBUSY; -		goto err_busy; -	} - -	imx_drm_connector = kzalloc(sizeof(*imx_drm_connector), GFP_KERNEL); -	if (!imx_drm_connector) { -		ret = -ENOMEM; -		goto err_alloc; -	} - -	imx_drm_connector->connector = connector; -	imx_drm_connector->owner = owner; - -	ret = imx_drm_connector_register(imx_drm_connector); -	if (ret) -		goto err_register; - -	list_add_tail(&imx_drm_connector->list, &imxdrm->connector_list); - -	*new_con = imx_drm_connector; - -	mutex_unlock(&imxdrm->mutex); - -	return 0; - -err_register: -	kfree(imx_drm_connector); -err_alloc: -err_busy: -	mutex_unlock(&imxdrm->mutex); - -	return ret; -} -EXPORT_SYMBOL_GPL(imx_drm_add_connector); - -void imx_drm_fb_helper_set(struct drm_fbdev_cma *fbdev_helper) -{ -	struct imx_drm_device *imxdrm = __imx_drm_device(); - -	imxdrm->fbhelper = fbdev_helper; -} -EXPORT_SYMBOL_GPL(imx_drm_fb_helper_set); - -/* - * imx_drm_remove_connector - remove a connector - */ -int imx_drm_remove_connector(struct imx_drm_connector *imx_drm_connector) -{ -	struct imx_drm_device *imxdrm = __imx_drm_device(); - -	mutex_lock(&imxdrm->mutex); - -	imx_drm_connector_unregister(imx_drm_connector); - -	list_del(&imx_drm_connector->list); +	if (!node || !imx_crtc) +		return -EINVAL; -	mutex_unlock(&imxdrm->mutex); +	do { +		ep = imx_drm_of_get_next_endpoint(node, ep); +		if (!ep) +			break; -	kfree(imx_drm_connector); +		port = of_graph_get_remote_port(ep); +		of_node_put(port); +		if (port == imx_crtc->port) { +			ret = of_graph_parse_endpoint(ep, &endpoint); +			return ret ? ret : endpoint.port; +		} +	} while (ep); -	return 0; +	return -EINVAL;  } -EXPORT_SYMBOL_GPL(imx_drm_remove_connector); +EXPORT_SYMBOL_GPL(imx_drm_encoder_get_mux_id);  static const struct drm_ioctl_desc imx_drm_ioctls[] = {  	/* none so far */  };  static struct drm_driver imx_drm_driver = { -	.driver_features	= DRIVER_MODESET | DRIVER_GEM, +	.driver_features	= DRIVER_MODESET | DRIVER_GEM | DRIVER_PRIME,  	.load			= imx_drm_driver_load,  	.unload			= imx_drm_driver_unload,  	.lastclose		= imx_drm_driver_lastclose, +	.preclose		= imx_drm_driver_preclose,  	.gem_free_object	= drm_gem_cma_free_object,  	.gem_vm_ops		= &drm_gem_cma_vm_ops,  	.dumb_create		= drm_gem_cma_dumb_create,  	.dumb_map_offset	= drm_gem_cma_dumb_map_offset,  	.dumb_destroy		= drm_gem_dumb_destroy, +	.prime_handle_to_fd	= drm_gem_prime_handle_to_fd, +	.prime_fd_to_handle	= drm_gem_prime_fd_to_handle, +	.gem_prime_import	= drm_gem_prime_import, +	.gem_prime_export	= drm_gem_prime_export, +	.gem_prime_get_sg_table	= drm_gem_cma_prime_get_sg_table, +	.gem_prime_import_sg_table = drm_gem_cma_prime_import_sg_table, +	.gem_prime_vmap		= drm_gem_cma_prime_vmap, +	.gem_prime_vunmap	= drm_gem_cma_prime_vunmap, +	.gem_prime_mmap		= drm_gem_cma_prime_mmap,  	.get_vblank_counter	= drm_vblank_count,  	.enable_vblank		= imx_drm_enable_vblank,  	.disable_vblank		= imx_drm_disable_vblank, @@ -803,76 +557,161 @@ static struct drm_driver imx_drm_driver = {  	.patchlevel		= 0,  }; -static int imx_drm_platform_probe(struct platform_device *pdev) +static int compare_of(struct device *dev, void *data)  { -	imx_drm_device->dev = &pdev->dev; +	struct device_node *np = data; + +	/* Special case for LDB, one device for two channels */ +	if (of_node_cmp(np->name, "lvds-channel") == 0) { +		np = of_get_parent(np); +		of_node_put(np); +	} -	return drm_platform_init(&imx_drm_driver, pdev); +	return dev->of_node == np;  } -static int imx_drm_platform_remove(struct platform_device *pdev) +static LIST_HEAD(imx_drm_components); + +static int imx_drm_add_components(struct device *master, struct master *m)  { -	drm_platform_exit(&imx_drm_driver, pdev); +	struct imx_drm_component *component; +	int ret; +	list_for_each_entry(component, &imx_drm_components, list) { +		ret = component_master_add_child(m, compare_of, +						 component->of_node); +		if (ret) +			return ret; +	}  	return 0;  } -static struct platform_driver imx_drm_pdrv = { -	.probe		= imx_drm_platform_probe, -	.remove		= imx_drm_platform_remove, -	.driver		= { -		.owner	= THIS_MODULE, -		.name	= "imx-drm", -	}, +static int imx_drm_bind(struct device *dev) +{ +	return drm_platform_init(&imx_drm_driver, to_platform_device(dev)); +} + +static void imx_drm_unbind(struct device *dev) +{ +	drm_put_dev(dev_get_drvdata(dev)); +} + +static const struct component_master_ops imx_drm_ops = { +	.add_components = imx_drm_add_components, +	.bind = imx_drm_bind, +	.unbind = imx_drm_unbind,  }; -static struct platform_device *imx_drm_pdev; +static struct imx_drm_component *imx_drm_find_component(struct device *dev, +		struct device_node *node) +{ +	struct imx_drm_component *component; -static int __init imx_drm_init(void) +	list_for_each_entry(component, &imx_drm_components, list) +		if (component->of_node == node) +			return component; + +	return NULL; +} + +static int imx_drm_add_component(struct device *dev, struct device_node *node)  { -	int ret; +	struct imx_drm_component *component; + +	if (imx_drm_find_component(dev, node)) +		return 0; -	imx_drm_device = kzalloc(sizeof(*imx_drm_device), GFP_KERNEL); -	if (!imx_drm_device) +	component = devm_kzalloc(dev, sizeof(*component), GFP_KERNEL); +	if (!component)  		return -ENOMEM; -	mutex_init(&imx_drm_device->mutex); -	INIT_LIST_HEAD(&imx_drm_device->crtc_list); -	INIT_LIST_HEAD(&imx_drm_device->connector_list); -	INIT_LIST_HEAD(&imx_drm_device->encoder_list); +	component->of_node = node; +	list_add_tail(&component->list, &imx_drm_components); -	imx_drm_pdev = platform_device_register_simple("imx-drm", -1, NULL, 0); -	if (!imx_drm_pdev) { -		ret = -EINVAL; -		goto err_pdev; +	return 0; +} + +static int imx_drm_platform_probe(struct platform_device *pdev) +{ +	struct device_node *ep, *port, *remote; +	int ret; +	int i; + +	/* +	 * Bind the IPU display interface ports first, so that +	 * imx_drm_encoder_parse_of called from encoder .bind callbacks +	 * works as expected. +	 */ +	for (i = 0; ; i++) { +		port = of_parse_phandle(pdev->dev.of_node, "ports", i); +		if (!port) +			break; + +		ret = imx_drm_add_component(&pdev->dev, port); +		if (ret < 0) +			return ret;  	} -	imx_drm_pdev->dev.coherent_dma_mask = DMA_BIT_MASK(32), +	if (i == 0) { +		dev_err(&pdev->dev, "missing 'ports' property\n"); +		return -ENODEV; +	} -	ret = platform_driver_register(&imx_drm_pdrv); -	if (ret) -		goto err_pdrv; +	/* Then bind all encoders */ +	for (i = 0; ; i++) { +		port = of_parse_phandle(pdev->dev.of_node, "ports", i); +		if (!port) +			break; -	return 0; +		for_each_child_of_node(port, ep) { +			remote = of_graph_get_remote_port_parent(ep); +			if (!remote || !of_device_is_available(remote)) { +				of_node_put(remote); +				continue; +			} else if (!of_device_is_available(remote->parent)) { +				dev_warn(&pdev->dev, "parent device of %s is not available\n", +					 remote->full_name); +				of_node_put(remote); +				continue; +			} -err_pdrv: -	platform_device_unregister(imx_drm_pdev); -err_pdev: -	kfree(imx_drm_device); +			ret = imx_drm_add_component(&pdev->dev, remote); +			of_node_put(remote); +			if (ret < 0) +				return ret; +		} +		of_node_put(port); +	} -	return ret; +	ret = dma_set_coherent_mask(&pdev->dev, DMA_BIT_MASK(32)); +	if (ret) +		return ret; + +	return component_master_add(&pdev->dev, &imx_drm_ops);  } -static void __exit imx_drm_exit(void) +static int imx_drm_platform_remove(struct platform_device *pdev)  { -	platform_device_unregister(imx_drm_pdev); -	platform_driver_unregister(&imx_drm_pdrv); - -	kfree(imx_drm_device); +	component_master_del(&pdev->dev, &imx_drm_ops); +	return 0;  } -module_init(imx_drm_init); -module_exit(imx_drm_exit); +static const struct of_device_id imx_drm_dt_ids[] = { +	{ .compatible = "fsl,imx-display-subsystem", }, +	{ /* sentinel */ }, +}; +MODULE_DEVICE_TABLE(of, imx_drm_dt_ids); + +static struct platform_driver imx_drm_pdrv = { +	.probe		= imx_drm_platform_probe, +	.remove		= imx_drm_platform_remove, +	.driver		= { +		.owner	= THIS_MODULE, +		.name	= "imx-drm", +		.of_match_table = imx_drm_dt_ids, +	}, +}; +module_platform_driver(imx_drm_pdrv);  MODULE_AUTHOR("Sascha Hauer <s.hauer@pengutronix.de>");  MODULE_DESCRIPTION("i.MX drm driver core"); diff --git a/drivers/staging/imx-drm/imx-drm.h b/drivers/staging/imx-drm/imx-drm.h index f2aac91ddf5..7453ae00c41 100644 --- a/drivers/staging/imx-drm/imx-drm.h +++ b/drivers/staging/imx-drm/imx-drm.h @@ -1,19 +1,19 @@  #ifndef _IMX_DRM_H_  #define _IMX_DRM_H_ -#include <linux/videodev2.h> - -#define IPU_PIX_FMT_GBR24	v4l2_fourcc('G', 'B', 'R', '3') - +struct device_node;  struct drm_crtc;  struct drm_connector;  struct drm_device; +struct drm_display_mode;  struct drm_encoder; -struct imx_drm_crtc;  struct drm_fbdev_cma;  struct drm_framebuffer; +struct imx_drm_crtc;  struct platform_device; +int imx_drm_crtc_id(struct imx_drm_crtc *crtc); +  struct imx_drm_crtc_helper_funcs {  	int (*enable_vblank)(struct drm_crtc *crtc);  	void (*disable_vblank)(struct drm_crtc *crtc); @@ -23,10 +23,10 @@ struct imx_drm_crtc_helper_funcs {  	const struct drm_crtc_funcs *crtc_funcs;  }; -int imx_drm_add_crtc(struct drm_crtc *crtc, +int imx_drm_add_crtc(struct drm_device *drm, struct drm_crtc *crtc,  		struct imx_drm_crtc **new_crtc,  		const struct imx_drm_crtc_helper_funcs *imx_helper_funcs, -		struct module *owner, void *cookie, int id); +		struct device_node *port);  int imx_drm_remove_crtc(struct imx_drm_crtc *);  int imx_drm_init_drm(struct platform_device *pdev,  		int preferred_bpp); @@ -36,35 +36,21 @@ int imx_drm_crtc_vblank_get(struct imx_drm_crtc *imx_drm_crtc);  void imx_drm_crtc_vblank_put(struct imx_drm_crtc *imx_drm_crtc);  void imx_drm_handle_vblank(struct imx_drm_crtc *imx_drm_crtc); -struct imx_drm_encoder; -int imx_drm_add_encoder(struct drm_encoder *encoder, -		struct imx_drm_encoder **new_enc, -		struct module *owner); -int imx_drm_remove_encoder(struct imx_drm_encoder *); - -struct imx_drm_connector; -int imx_drm_add_connector(struct drm_connector *connector, -		struct imx_drm_connector **new_con, -		struct module *owner); -int imx_drm_remove_connector(struct imx_drm_connector *); -  void imx_drm_mode_config_init(struct drm_device *drm);  struct drm_gem_cma_object *imx_drm_fb_get_obj(struct drm_framebuffer *fb); -struct drm_device *imx_drm_device_get(void); -void imx_drm_device_put(void); -int imx_drm_crtc_panel_format_pins(struct drm_crtc *crtc, u32 encoder_type, +int imx_drm_panel_format_pins(struct drm_encoder *encoder,  		u32 interface_pix_fmt, int hsync_pin, int vsync_pin); -int imx_drm_crtc_panel_format(struct drm_crtc *crtc, u32 encoder_type, +int imx_drm_panel_format(struct drm_encoder *encoder,  		u32 interface_pix_fmt); -void imx_drm_fb_helper_set(struct drm_fbdev_cma *fbdev_helper); -struct device_node; +int imx_drm_encoder_get_mux_id(struct device_node *node, +		struct drm_encoder *encoder); +int imx_drm_encoder_parse_of(struct drm_device *drm, +	struct drm_encoder *encoder, struct device_node *np); -int imx_drm_encoder_get_mux_id(struct imx_drm_encoder *imx_drm_encoder, -		struct drm_crtc *crtc); -int imx_drm_encoder_add_possible_crtcs(struct imx_drm_encoder *imx_drm_encoder, -		struct device_node *np); +void imx_drm_connector_destroy(struct drm_connector *connector); +void imx_drm_encoder_destroy(struct drm_encoder *encoder);  #endif /* _IMX_DRM_H_ */ diff --git a/drivers/staging/imx-drm/imx-fb.c b/drivers/staging/imx-drm/imx-fb.c deleted file mode 100644 index 03a7b4e14f6..00000000000 --- a/drivers/staging/imx-drm/imx-fb.c +++ /dev/null @@ -1,47 +0,0 @@ -/* - * i.MX drm driver - * - * Copyright (C) 2012 Sascha Hauer, Pengutronix - * - * Based on Samsung Exynos code - * - * Copyright (c) 2011 Samsung Electronics Co., Ltd. - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * as published by the Free Software Foundation; either version 2 - * of the License, or (at your option) any later version. - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the - * GNU General Public License for more details. - * - */ -#include <linux/module.h> -#include <drm/drmP.h> -#include <drm/drm_crtc.h> -#include <drm/drm_crtc_helper.h> -#include <drm/drm_gem_cma_helper.h> -#include <drm/drm_fb_cma_helper.h> - -#include "imx-drm.h" - -static struct drm_mode_config_funcs imx_drm_mode_config_funcs = { -	.fb_create = drm_fb_cma_create, -}; - -void imx_drm_mode_config_init(struct drm_device *dev) -{ -	dev->mode_config.min_width = 64; -	dev->mode_config.min_height = 64; - -	/* -	 * set max width and height as default value(4096x4096). -	 * this value would be used to check framebuffer size limitation -	 * at drm_mode_addfb(). -	 */ -	dev->mode_config.max_width = 4096; -	dev->mode_config.max_height = 4096; - -	dev->mode_config.funcs = &imx_drm_mode_config_funcs; -} diff --git a/drivers/staging/imx-drm/imx-fbdev.c b/drivers/staging/imx-drm/imx-fbdev.c deleted file mode 100644 index 8331739c3d0..00000000000 --- a/drivers/staging/imx-drm/imx-fbdev.c +++ /dev/null @@ -1,74 +0,0 @@ -/* - * i.MX drm driver - * - * Copyright (C) 2012 Sascha Hauer, Pengutronix - * - * Based on Samsung Exynos code - * - * Copyright (c) 2011 Samsung Electronics Co., Ltd. - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * as published by the Free Software Foundation; either version 2 - * of the License, or (at your option) any later version. - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the - * GNU General Public License for more details. - * - */ -#include <linux/module.h> -#include <drm/drmP.h> -#include <drm/drm_crtc.h> -#include <drm/drm_crtc_helper.h> -#include <drm/drm_fb_cma_helper.h> - -#include "imx-drm.h" - -#define MAX_CONNECTOR		4 -#define PREFERRED_BPP		16 - -static struct drm_fbdev_cma *fbdev_cma; - -static int legacyfb_depth = 16; - -module_param(legacyfb_depth, int, 0444); - -static int __init imx_fb_helper_init(void) -{ -	struct drm_device *drm = imx_drm_device_get(); - -	if (!drm) -		return -EINVAL; - -	if (legacyfb_depth != 16 && legacyfb_depth != 32) { -		pr_warn("i.MX legacyfb: invalid legacyfb_depth setting. defaulting to 16bpp\n"); -		legacyfb_depth = 16; -	} - -	fbdev_cma = drm_fbdev_cma_init(drm, legacyfb_depth, -			drm->mode_config.num_crtc, MAX_CONNECTOR); - -	if (IS_ERR(fbdev_cma)) { -		imx_drm_device_put(); -		return PTR_ERR(fbdev_cma); -	} - -	imx_drm_fb_helper_set(fbdev_cma); - -	return 0; -} - -static void __exit imx_fb_helper_exit(void) -{ -	imx_drm_fb_helper_set(NULL); -	drm_fbdev_cma_fini(fbdev_cma); -	imx_drm_device_put(); -} - -late_initcall(imx_fb_helper_init); -module_exit(imx_fb_helper_exit); - -MODULE_DESCRIPTION("Freescale i.MX legacy fb driver"); -MODULE_AUTHOR("Sascha Hauer, Pengutronix"); -MODULE_LICENSE("GPL"); diff --git a/drivers/staging/imx-drm/imx-hdmi.c b/drivers/staging/imx-drm/imx-hdmi.c new file mode 100644 index 00000000000..18c9ccd460b --- /dev/null +++ b/drivers/staging/imx-drm/imx-hdmi.c @@ -0,0 +1,1768 @@ +/* + * Copyright (C) 2011-2013 Freescale Semiconductor, Inc. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * SH-Mobile High-Definition Multimedia Interface (HDMI) driver + * for SLISHDMI13T and SLIPHDMIT IP cores + * + * Copyright (C) 2010, Guennadi Liakhovetski <g.liakhovetski@gmx.de> + */ + +#include <linux/component.h> +#include <linux/irq.h> +#include <linux/delay.h> +#include <linux/err.h> +#include <linux/clk.h> +#include <linux/hdmi.h> +#include <linux/regmap.h> +#include <linux/mfd/syscon.h> +#include <linux/mfd/syscon/imx6q-iomuxc-gpr.h> +#include <linux/of_device.h> + +#include <drm/drmP.h> +#include <drm/drm_crtc_helper.h> +#include <drm/drm_edid.h> +#include <drm/drm_encoder_slave.h> +#include <video/imx-ipu-v3.h> + +#include "imx-hdmi.h" +#include "imx-drm.h" + +#define HDMI_EDID_LEN		512 + +#define RGB			0 +#define YCBCR444		1 +#define YCBCR422_16BITS		2 +#define YCBCR422_8BITS		3 +#define XVYCC444		4 + +enum hdmi_datamap { +	RGB444_8B = 0x01, +	RGB444_10B = 0x03, +	RGB444_12B = 0x05, +	RGB444_16B = 0x07, +	YCbCr444_8B = 0x09, +	YCbCr444_10B = 0x0B, +	YCbCr444_12B = 0x0D, +	YCbCr444_16B = 0x0F, +	YCbCr422_8B = 0x16, +	YCbCr422_10B = 0x14, +	YCbCr422_12B = 0x12, +}; + +enum imx_hdmi_devtype { +	IMX6Q_HDMI, +	IMX6DL_HDMI, +}; + +static const u16 csc_coeff_default[3][4] = { +	{ 0x2000, 0x0000, 0x0000, 0x0000 }, +	{ 0x0000, 0x2000, 0x0000, 0x0000 }, +	{ 0x0000, 0x0000, 0x2000, 0x0000 } +}; + +static const u16 csc_coeff_rgb_out_eitu601[3][4] = { +	{ 0x2000, 0x6926, 0x74fd, 0x010e }, +	{ 0x2000, 0x2cdd, 0x0000, 0x7e9a }, +	{ 0x2000, 0x0000, 0x38b4, 0x7e3b } +}; + +static const u16 csc_coeff_rgb_out_eitu709[3][4] = { +	{ 0x2000, 0x7106, 0x7a02, 0x00a7 }, +	{ 0x2000, 0x3264, 0x0000, 0x7e6d }, +	{ 0x2000, 0x0000, 0x3b61, 0x7e25 } +}; + +static const u16 csc_coeff_rgb_in_eitu601[3][4] = { +	{ 0x2591, 0x1322, 0x074b, 0x0000 }, +	{ 0x6535, 0x2000, 0x7acc, 0x0200 }, +	{ 0x6acd, 0x7534, 0x2000, 0x0200 } +}; + +static const u16 csc_coeff_rgb_in_eitu709[3][4] = { +	{ 0x2dc5, 0x0d9b, 0x049e, 0x0000 }, +	{ 0x62f0, 0x2000, 0x7d11, 0x0200 }, +	{ 0x6756, 0x78ab, 0x2000, 0x0200 } +}; + +struct hdmi_vmode { +	bool mdvi; +	bool mhsyncpolarity; +	bool mvsyncpolarity; +	bool minterlaced; +	bool mdataenablepolarity; + +	unsigned int mpixelclock; +	unsigned int mpixelrepetitioninput; +	unsigned int mpixelrepetitionoutput; +}; + +struct hdmi_data_info { +	unsigned int enc_in_format; +	unsigned int enc_out_format; +	unsigned int enc_color_depth; +	unsigned int colorimetry; +	unsigned int pix_repet_factor; +	unsigned int hdcp_enable; +	struct hdmi_vmode video_mode; +}; + +struct imx_hdmi { +	struct drm_connector connector; +	struct drm_encoder encoder; + +	enum imx_hdmi_devtype dev_type; +	struct device *dev; +	struct clk *isfr_clk; +	struct clk *iahb_clk; + +	struct hdmi_data_info hdmi_data; +	int vic; + +	u8 edid[HDMI_EDID_LEN]; +	bool cable_plugin; + +	bool phy_enabled; +	struct drm_display_mode previous_mode; + +	struct regmap *regmap; +	struct i2c_adapter *ddc; +	void __iomem *regs; + +	unsigned int sample_rate; +	int ratio; +}; + +static void imx_hdmi_set_ipu_di_mux(struct imx_hdmi *hdmi, int ipu_di) +{ +	regmap_update_bits(hdmi->regmap, IOMUXC_GPR3, +			   IMX6Q_GPR3_HDMI_MUX_CTL_MASK, +			   ipu_di << IMX6Q_GPR3_HDMI_MUX_CTL_SHIFT); +} + +static inline void hdmi_writeb(struct imx_hdmi *hdmi, u8 val, int offset) +{ +	writeb(val, hdmi->regs + offset); +} + +static inline u8 hdmi_readb(struct imx_hdmi *hdmi, int offset) +{ +	return readb(hdmi->regs + offset); +} + +static void hdmi_modb(struct imx_hdmi *hdmi, u8 data, u8 mask, unsigned reg) +{ +	u8 val = hdmi_readb(hdmi, reg) & ~mask; + +	val |= data & mask; +	hdmi_writeb(hdmi, val, reg); +} + +static void hdmi_mask_writeb(struct imx_hdmi *hdmi, u8 data, unsigned int reg, +		      u8 shift, u8 mask) +{ +	hdmi_modb(hdmi, data << shift, mask, reg); +} + +static void hdmi_set_clock_regenerator_n(struct imx_hdmi *hdmi, +					 unsigned int value) +{ +	hdmi_writeb(hdmi, value & 0xff, HDMI_AUD_N1); +	hdmi_writeb(hdmi, (value >> 8) & 0xff, HDMI_AUD_N2); +	hdmi_writeb(hdmi, (value >> 16) & 0x0f, HDMI_AUD_N3); + +	/* nshift factor = 0 */ +	hdmi_modb(hdmi, 0, HDMI_AUD_CTS3_N_SHIFT_MASK, HDMI_AUD_CTS3); +} + +static void hdmi_regenerate_cts(struct imx_hdmi *hdmi, unsigned int cts) +{ +	/* Must be set/cleared first */ +	hdmi_modb(hdmi, 0, HDMI_AUD_CTS3_CTS_MANUAL, HDMI_AUD_CTS3); + +	hdmi_writeb(hdmi, cts & 0xff, HDMI_AUD_CTS1); +	hdmi_writeb(hdmi, (cts >> 8) & 0xff, HDMI_AUD_CTS2); +	hdmi_writeb(hdmi, ((cts >> 16) & HDMI_AUD_CTS3_AUDCTS19_16_MASK) | +		    HDMI_AUD_CTS3_CTS_MANUAL, HDMI_AUD_CTS3); +} + +static unsigned int hdmi_compute_n(unsigned int freq, unsigned long pixel_clk, +				   unsigned int ratio) +{ +	unsigned int n = (128 * freq) / 1000; + +	switch (freq) { +	case 32000: +		if (pixel_clk == 25170000) +			n = (ratio == 150) ? 9152 : 4576; +		else if (pixel_clk == 27020000) +			n = (ratio == 150) ? 8192 : 4096; +		else if (pixel_clk == 74170000 || pixel_clk == 148350000) +			n = 11648; +		else +			n = 4096; +		break; + +	case 44100: +		if (pixel_clk == 25170000) +			n = 7007; +		else if (pixel_clk == 74170000) +			n = 17836; +		else if (pixel_clk == 148350000) +			n = (ratio == 150) ? 17836 : 8918; +		else +			n = 6272; +		break; + +	case 48000: +		if (pixel_clk == 25170000) +			n = (ratio == 150) ? 9152 : 6864; +		else if (pixel_clk == 27020000) +			n = (ratio == 150) ? 8192 : 6144; +		else if (pixel_clk == 74170000) +			n = 11648; +		else if (pixel_clk == 148350000) +			n = (ratio == 150) ? 11648 : 5824; +		else +			n = 6144; +		break; + +	case 88200: +		n = hdmi_compute_n(44100, pixel_clk, ratio) * 2; +		break; + +	case 96000: +		n = hdmi_compute_n(48000, pixel_clk, ratio) * 2; +		break; + +	case 176400: +		n = hdmi_compute_n(44100, pixel_clk, ratio) * 4; +		break; + +	case 192000: +		n = hdmi_compute_n(48000, pixel_clk, ratio) * 4; +		break; + +	default: +		break; +	} + +	return n; +} + +static unsigned int hdmi_compute_cts(unsigned int freq, unsigned long pixel_clk, +				     unsigned int ratio) +{ +	unsigned int cts = 0; + +	pr_debug("%s: freq: %d pixel_clk: %ld ratio: %d\n", __func__, freq, +		 pixel_clk, ratio); + +	switch (freq) { +	case 32000: +		if (pixel_clk == 297000000) { +			cts = 222750; +			break; +		} +	case 48000: +	case 96000: +	case 192000: +		switch (pixel_clk) { +		case 25200000: +		case 27000000: +		case 54000000: +		case 74250000: +		case 148500000: +			cts = pixel_clk / 1000; +			break; +		case 297000000: +			cts = 247500; +			break; +		/* +		 * All other TMDS clocks are not supported by +		 * DWC_hdmi_tx. The TMDS clocks divided or +		 * multiplied by 1,001 coefficients are not +		 * supported. +		 */ +		default: +			break; +		} +		break; +	case 44100: +	case 88200: +	case 176400: +		switch (pixel_clk) { +		case 25200000: +			cts = 28000; +			break; +		case 27000000: +			cts = 30000; +			break; +		case 54000000: +			cts = 60000; +			break; +		case 74250000: +			cts = 82500; +			break; +		case 148500000: +			cts = 165000; +			break; +		case 297000000: +			cts = 247500; +			break; +		default: +			break; +		} +		break; +	default: +		break; +	} +	if (ratio == 100) +		return cts; +	else +		return (cts * ratio) / 100; +} + +static void hdmi_set_clk_regenerator(struct imx_hdmi *hdmi, +	unsigned long pixel_clk) +{ +	unsigned int clk_n, clk_cts; + +	clk_n = hdmi_compute_n(hdmi->sample_rate, pixel_clk, +			       hdmi->ratio); +	clk_cts = hdmi_compute_cts(hdmi->sample_rate, pixel_clk, +				   hdmi->ratio); + +	if (!clk_cts) { +		dev_dbg(hdmi->dev, "%s: pixel clock not supported: %lu\n", +			 __func__, pixel_clk); +		return; +	} + +	dev_dbg(hdmi->dev, "%s: samplerate=%d  ratio=%d  pixelclk=%lu  N=%d cts=%d\n", +		__func__, hdmi->sample_rate, hdmi->ratio, +		pixel_clk, clk_n, clk_cts); + +	hdmi_set_clock_regenerator_n(hdmi, clk_n); +	hdmi_regenerate_cts(hdmi, clk_cts); +} + +static void hdmi_init_clk_regenerator(struct imx_hdmi *hdmi) +{ +	hdmi_set_clk_regenerator(hdmi, 74250000); +} + +static void hdmi_clk_regenerator_update_pixel_clock(struct imx_hdmi *hdmi) +{ +	hdmi_set_clk_regenerator(hdmi, hdmi->hdmi_data.video_mode.mpixelclock); +} + +/* + * this submodule is responsible for the video data synchronization. + * for example, for RGB 4:4:4 input, the data map is defined as + *			pin{47~40} <==> R[7:0] + *			pin{31~24} <==> G[7:0] + *			pin{15~8}  <==> B[7:0] + */ +static void hdmi_video_sample(struct imx_hdmi *hdmi) +{ +	int color_format = 0; +	u8 val; + +	if (hdmi->hdmi_data.enc_in_format == RGB) { +		if (hdmi->hdmi_data.enc_color_depth == 8) +			color_format = 0x01; +		else if (hdmi->hdmi_data.enc_color_depth == 10) +			color_format = 0x03; +		else if (hdmi->hdmi_data.enc_color_depth == 12) +			color_format = 0x05; +		else if (hdmi->hdmi_data.enc_color_depth == 16) +			color_format = 0x07; +		else +			return; +	} else if (hdmi->hdmi_data.enc_in_format == YCBCR444) { +		if (hdmi->hdmi_data.enc_color_depth == 8) +			color_format = 0x09; +		else if (hdmi->hdmi_data.enc_color_depth == 10) +			color_format = 0x0B; +		else if (hdmi->hdmi_data.enc_color_depth == 12) +			color_format = 0x0D; +		else if (hdmi->hdmi_data.enc_color_depth == 16) +			color_format = 0x0F; +		else +			return; +	} else if (hdmi->hdmi_data.enc_in_format == YCBCR422_8BITS) { +		if (hdmi->hdmi_data.enc_color_depth == 8) +			color_format = 0x16; +		else if (hdmi->hdmi_data.enc_color_depth == 10) +			color_format = 0x14; +		else if (hdmi->hdmi_data.enc_color_depth == 12) +			color_format = 0x12; +		else +			return; +	} + +	val = HDMI_TX_INVID0_INTERNAL_DE_GENERATOR_DISABLE | +		((color_format << HDMI_TX_INVID0_VIDEO_MAPPING_OFFSET) & +		HDMI_TX_INVID0_VIDEO_MAPPING_MASK); +	hdmi_writeb(hdmi, val, HDMI_TX_INVID0); + +	/* Enable TX stuffing: When DE is inactive, fix the output data to 0 */ +	val = HDMI_TX_INSTUFFING_BDBDATA_STUFFING_ENABLE | +		HDMI_TX_INSTUFFING_RCRDATA_STUFFING_ENABLE | +		HDMI_TX_INSTUFFING_GYDATA_STUFFING_ENABLE; +	hdmi_writeb(hdmi, val, HDMI_TX_INSTUFFING); +	hdmi_writeb(hdmi, 0x0, HDMI_TX_GYDATA0); +	hdmi_writeb(hdmi, 0x0, HDMI_TX_GYDATA1); +	hdmi_writeb(hdmi, 0x0, HDMI_TX_RCRDATA0); +	hdmi_writeb(hdmi, 0x0, HDMI_TX_RCRDATA1); +	hdmi_writeb(hdmi, 0x0, HDMI_TX_BCBDATA0); +	hdmi_writeb(hdmi, 0x0, HDMI_TX_BCBDATA1); +} + +static int is_color_space_conversion(struct imx_hdmi *hdmi) +{ +	return hdmi->hdmi_data.enc_in_format != hdmi->hdmi_data.enc_out_format; +} + +static int is_color_space_decimation(struct imx_hdmi *hdmi) +{ +	if (hdmi->hdmi_data.enc_out_format != YCBCR422_8BITS) +		return 0; +	if (hdmi->hdmi_data.enc_in_format == RGB || +	    hdmi->hdmi_data.enc_in_format == YCBCR444) +		return 1; +	return 0; +} + +static int is_color_space_interpolation(struct imx_hdmi *hdmi) +{ +	if (hdmi->hdmi_data.enc_in_format != YCBCR422_8BITS) +		return 0; +	if (hdmi->hdmi_data.enc_out_format == RGB || +	    hdmi->hdmi_data.enc_out_format == YCBCR444) +		return 1; +	return 0; +} + +static void imx_hdmi_update_csc_coeffs(struct imx_hdmi *hdmi) +{ +	const u16 (*csc_coeff)[3][4] = &csc_coeff_default; +	unsigned i; +	u32 csc_scale = 1; + +	if (is_color_space_conversion(hdmi)) { +		if (hdmi->hdmi_data.enc_out_format == RGB) { +			if (hdmi->hdmi_data.colorimetry == +					HDMI_COLORIMETRY_ITU_601) +				csc_coeff = &csc_coeff_rgb_out_eitu601; +			else +				csc_coeff = &csc_coeff_rgb_out_eitu709; +		} else if (hdmi->hdmi_data.enc_in_format == RGB) { +			if (hdmi->hdmi_data.colorimetry == +					HDMI_COLORIMETRY_ITU_601) +				csc_coeff = &csc_coeff_rgb_in_eitu601; +			else +				csc_coeff = &csc_coeff_rgb_in_eitu709; +			csc_scale = 0; +		} +	} + +	/* The CSC registers are sequential, alternating MSB then LSB */ +	for (i = 0; i < ARRAY_SIZE(csc_coeff_default[0]); i++) { +		u16 coeff_a = (*csc_coeff)[0][i]; +		u16 coeff_b = (*csc_coeff)[1][i]; +		u16 coeff_c = (*csc_coeff)[2][i]; + +		hdmi_writeb(hdmi, coeff_a & 0xff, +			HDMI_CSC_COEF_A1_LSB + i * 2); +		hdmi_writeb(hdmi, coeff_a >> 8, HDMI_CSC_COEF_A1_MSB + i * 2); +		hdmi_writeb(hdmi, coeff_b & 0xff, HDMI_CSC_COEF_B1_LSB + i * 2); +		hdmi_writeb(hdmi, coeff_b >> 8, HDMI_CSC_COEF_B1_MSB + i * 2); +		hdmi_writeb(hdmi, coeff_c & 0xff, +			HDMI_CSC_COEF_C1_LSB + i * 2); +		hdmi_writeb(hdmi, coeff_c >> 8, HDMI_CSC_COEF_C1_MSB + i * 2); +	} + +	hdmi_modb(hdmi, csc_scale, HDMI_CSC_SCALE_CSCSCALE_MASK, +		  HDMI_CSC_SCALE); +} + +static void hdmi_video_csc(struct imx_hdmi *hdmi) +{ +	int color_depth = 0; +	int interpolation = HDMI_CSC_CFG_INTMODE_DISABLE; +	int decimation = 0; + +	/* YCC422 interpolation to 444 mode */ +	if (is_color_space_interpolation(hdmi)) +		interpolation = HDMI_CSC_CFG_INTMODE_CHROMA_INT_FORMULA1; +	else if (is_color_space_decimation(hdmi)) +		decimation = HDMI_CSC_CFG_DECMODE_CHROMA_INT_FORMULA3; + +	if (hdmi->hdmi_data.enc_color_depth == 8) +		color_depth = HDMI_CSC_SCALE_CSC_COLORDE_PTH_24BPP; +	else if (hdmi->hdmi_data.enc_color_depth == 10) +		color_depth = HDMI_CSC_SCALE_CSC_COLORDE_PTH_30BPP; +	else if (hdmi->hdmi_data.enc_color_depth == 12) +		color_depth = HDMI_CSC_SCALE_CSC_COLORDE_PTH_36BPP; +	else if (hdmi->hdmi_data.enc_color_depth == 16) +		color_depth = HDMI_CSC_SCALE_CSC_COLORDE_PTH_48BPP; +	else +		return; + +	/* Configure the CSC registers */ +	hdmi_writeb(hdmi, interpolation | decimation, HDMI_CSC_CFG); +	hdmi_modb(hdmi, color_depth, HDMI_CSC_SCALE_CSC_COLORDE_PTH_MASK, +		  HDMI_CSC_SCALE); + +	imx_hdmi_update_csc_coeffs(hdmi); +} + +/* + * HDMI video packetizer is used to packetize the data. + * for example, if input is YCC422 mode or repeater is used, + * data should be repacked this module can be bypassed. + */ +static void hdmi_video_packetize(struct imx_hdmi *hdmi) +{ +	unsigned int color_depth = 0; +	unsigned int remap_size = HDMI_VP_REMAP_YCC422_16bit; +	unsigned int output_select = HDMI_VP_CONF_OUTPUT_SELECTOR_PP; +	struct hdmi_data_info *hdmi_data = &hdmi->hdmi_data; +	u8 val, vp_conf; + +	if (hdmi_data->enc_out_format == RGB +		|| hdmi_data->enc_out_format == YCBCR444) { +		if (!hdmi_data->enc_color_depth) +			output_select = HDMI_VP_CONF_OUTPUT_SELECTOR_BYPASS; +		else if (hdmi_data->enc_color_depth == 8) { +			color_depth = 4; +			output_select = HDMI_VP_CONF_OUTPUT_SELECTOR_BYPASS; +		} else if (hdmi_data->enc_color_depth == 10) +			color_depth = 5; +		else if (hdmi_data->enc_color_depth == 12) +			color_depth = 6; +		else if (hdmi_data->enc_color_depth == 16) +			color_depth = 7; +		else +			return; +	} else if (hdmi_data->enc_out_format == YCBCR422_8BITS) { +		if (!hdmi_data->enc_color_depth || +		    hdmi_data->enc_color_depth == 8) +			remap_size = HDMI_VP_REMAP_YCC422_16bit; +		else if (hdmi_data->enc_color_depth == 10) +			remap_size = HDMI_VP_REMAP_YCC422_20bit; +		else if (hdmi_data->enc_color_depth == 12) +			remap_size = HDMI_VP_REMAP_YCC422_24bit; +		else +			return; +		output_select = HDMI_VP_CONF_OUTPUT_SELECTOR_YCC422; +	} else +		return; + +	/* set the packetizer registers */ +	val = ((color_depth << HDMI_VP_PR_CD_COLOR_DEPTH_OFFSET) & +		HDMI_VP_PR_CD_COLOR_DEPTH_MASK) | +		((hdmi_data->pix_repet_factor << +		HDMI_VP_PR_CD_DESIRED_PR_FACTOR_OFFSET) & +		HDMI_VP_PR_CD_DESIRED_PR_FACTOR_MASK); +	hdmi_writeb(hdmi, val, HDMI_VP_PR_CD); + +	hdmi_modb(hdmi, HDMI_VP_STUFF_PR_STUFFING_STUFFING_MODE, +		  HDMI_VP_STUFF_PR_STUFFING_MASK, HDMI_VP_STUFF); + +	/* Data from pixel repeater block */ +	if (hdmi_data->pix_repet_factor > 1) { +		vp_conf = HDMI_VP_CONF_PR_EN_ENABLE | +			  HDMI_VP_CONF_BYPASS_SELECT_PIX_REPEATER; +	} else { /* data from packetizer block */ +		vp_conf = HDMI_VP_CONF_PR_EN_DISABLE | +			  HDMI_VP_CONF_BYPASS_SELECT_VID_PACKETIZER; +	} + +	hdmi_modb(hdmi, vp_conf, +		  HDMI_VP_CONF_PR_EN_MASK | +		  HDMI_VP_CONF_BYPASS_SELECT_MASK, HDMI_VP_CONF); + +	hdmi_modb(hdmi, 1 << HDMI_VP_STUFF_IDEFAULT_PHASE_OFFSET, +		  HDMI_VP_STUFF_IDEFAULT_PHASE_MASK, HDMI_VP_STUFF); + +	hdmi_writeb(hdmi, remap_size, HDMI_VP_REMAP); + +	if (output_select == HDMI_VP_CONF_OUTPUT_SELECTOR_PP) { +		vp_conf = HDMI_VP_CONF_BYPASS_EN_DISABLE | +			  HDMI_VP_CONF_PP_EN_ENABLE | +			  HDMI_VP_CONF_YCC422_EN_DISABLE; +	} else if (output_select == HDMI_VP_CONF_OUTPUT_SELECTOR_YCC422) { +		vp_conf = HDMI_VP_CONF_BYPASS_EN_DISABLE | +			  HDMI_VP_CONF_PP_EN_DISABLE | +			  HDMI_VP_CONF_YCC422_EN_ENABLE; +	} else if (output_select == HDMI_VP_CONF_OUTPUT_SELECTOR_BYPASS) { +		vp_conf = HDMI_VP_CONF_BYPASS_EN_ENABLE | +			  HDMI_VP_CONF_PP_EN_DISABLE | +			  HDMI_VP_CONF_YCC422_EN_DISABLE; +	} else { +		return; +	} + +	hdmi_modb(hdmi, vp_conf, +		  HDMI_VP_CONF_BYPASS_EN_MASK | HDMI_VP_CONF_PP_EN_ENMASK | +		  HDMI_VP_CONF_YCC422_EN_MASK, HDMI_VP_CONF); + +	hdmi_modb(hdmi, HDMI_VP_STUFF_PP_STUFFING_STUFFING_MODE | +			HDMI_VP_STUFF_YCC422_STUFFING_STUFFING_MODE, +		  HDMI_VP_STUFF_PP_STUFFING_MASK | +		  HDMI_VP_STUFF_YCC422_STUFFING_MASK, HDMI_VP_STUFF); + +	hdmi_modb(hdmi, output_select, HDMI_VP_CONF_OUTPUT_SELECTOR_MASK, +		  HDMI_VP_CONF); +} + +static inline void hdmi_phy_test_clear(struct imx_hdmi *hdmi, +						unsigned char bit) +{ +	hdmi_modb(hdmi, bit << HDMI_PHY_TST0_TSTCLR_OFFSET, +		  HDMI_PHY_TST0_TSTCLR_MASK, HDMI_PHY_TST0); +} + +static inline void hdmi_phy_test_enable(struct imx_hdmi *hdmi, +						unsigned char bit) +{ +	hdmi_modb(hdmi, bit << HDMI_PHY_TST0_TSTEN_OFFSET, +		  HDMI_PHY_TST0_TSTEN_MASK, HDMI_PHY_TST0); +} + +static inline void hdmi_phy_test_clock(struct imx_hdmi *hdmi, +						unsigned char bit) +{ +	hdmi_modb(hdmi, bit << HDMI_PHY_TST0_TSTCLK_OFFSET, +		  HDMI_PHY_TST0_TSTCLK_MASK, HDMI_PHY_TST0); +} + +static inline void hdmi_phy_test_din(struct imx_hdmi *hdmi, +						unsigned char bit) +{ +	hdmi_writeb(hdmi, bit, HDMI_PHY_TST1); +} + +static inline void hdmi_phy_test_dout(struct imx_hdmi *hdmi, +						unsigned char bit) +{ +	hdmi_writeb(hdmi, bit, HDMI_PHY_TST2); +} + +static bool hdmi_phy_wait_i2c_done(struct imx_hdmi *hdmi, int msec) +{ +	while ((hdmi_readb(hdmi, HDMI_IH_I2CMPHY_STAT0) & 0x3) == 0) { +		if (msec-- == 0) +			return false; +		udelay(1000); +	} +	return true; +} + +static void __hdmi_phy_i2c_write(struct imx_hdmi *hdmi, unsigned short data, +			      unsigned char addr) +{ +	hdmi_writeb(hdmi, 0xFF, HDMI_IH_I2CMPHY_STAT0); +	hdmi_writeb(hdmi, addr, HDMI_PHY_I2CM_ADDRESS_ADDR); +	hdmi_writeb(hdmi, (unsigned char)(data >> 8), +		HDMI_PHY_I2CM_DATAO_1_ADDR); +	hdmi_writeb(hdmi, (unsigned char)(data >> 0), +		HDMI_PHY_I2CM_DATAO_0_ADDR); +	hdmi_writeb(hdmi, HDMI_PHY_I2CM_OPERATION_ADDR_WRITE, +		HDMI_PHY_I2CM_OPERATION_ADDR); +	hdmi_phy_wait_i2c_done(hdmi, 1000); +} + +static int hdmi_phy_i2c_write(struct imx_hdmi *hdmi, unsigned short data, +				     unsigned char addr) +{ +	__hdmi_phy_i2c_write(hdmi, data, addr); +	return 0; +} + +static void imx_hdmi_phy_enable_power(struct imx_hdmi *hdmi, u8 enable) +{ +	hdmi_mask_writeb(hdmi, enable, HDMI_PHY_CONF0, +			 HDMI_PHY_CONF0_PDZ_OFFSET, +			 HDMI_PHY_CONF0_PDZ_MASK); +} + +static void imx_hdmi_phy_enable_tmds(struct imx_hdmi *hdmi, u8 enable) +{ +	hdmi_mask_writeb(hdmi, enable, HDMI_PHY_CONF0, +			 HDMI_PHY_CONF0_ENTMDS_OFFSET, +			 HDMI_PHY_CONF0_ENTMDS_MASK); +} + +static void imx_hdmi_phy_gen2_pddq(struct imx_hdmi *hdmi, u8 enable) +{ +	hdmi_mask_writeb(hdmi, enable, HDMI_PHY_CONF0, +			 HDMI_PHY_CONF0_GEN2_PDDQ_OFFSET, +			 HDMI_PHY_CONF0_GEN2_PDDQ_MASK); +} + +static void imx_hdmi_phy_gen2_txpwron(struct imx_hdmi *hdmi, u8 enable) +{ +	hdmi_mask_writeb(hdmi, enable, HDMI_PHY_CONF0, +			 HDMI_PHY_CONF0_GEN2_TXPWRON_OFFSET, +			 HDMI_PHY_CONF0_GEN2_TXPWRON_MASK); +} + +static void imx_hdmi_phy_sel_data_en_pol(struct imx_hdmi *hdmi, u8 enable) +{ +	hdmi_mask_writeb(hdmi, enable, HDMI_PHY_CONF0, +			 HDMI_PHY_CONF0_SELDATAENPOL_OFFSET, +			 HDMI_PHY_CONF0_SELDATAENPOL_MASK); +} + +static void imx_hdmi_phy_sel_interface_control(struct imx_hdmi *hdmi, u8 enable) +{ +	hdmi_mask_writeb(hdmi, enable, HDMI_PHY_CONF0, +			 HDMI_PHY_CONF0_SELDIPIF_OFFSET, +			 HDMI_PHY_CONF0_SELDIPIF_MASK); +} + +enum { +	RES_8, +	RES_10, +	RES_12, +	RES_MAX, +}; + +struct mpll_config { +	unsigned long mpixelclock; +	struct { +		u16 cpce; +		u16 gmp; +	} res[RES_MAX]; +}; + +static const struct mpll_config mpll_config[] = { +	{ +		45250000, { +			{ 0x01e0, 0x0000 }, +			{ 0x21e1, 0x0000 }, +			{ 0x41e2, 0x0000 } +		}, +	}, { +		92500000, { +			{ 0x0140, 0x0005 }, +			{ 0x2141, 0x0005 }, +			{ 0x4142, 0x0005 }, +		}, +	}, { +		148500000, { +			{ 0x00a0, 0x000a }, +			{ 0x20a1, 0x000a }, +			{ 0x40a2, 0x000a }, +		}, +	}, { +		~0UL, { +			{ 0x00a0, 0x000a }, +			{ 0x2001, 0x000f }, +			{ 0x4002, 0x000f }, +		}, +	} +}; + +struct curr_ctrl { +	unsigned long mpixelclock; +	u16 curr[RES_MAX]; +}; + +static const struct curr_ctrl curr_ctrl[] = { +	/*	pixelclk     bpp8    bpp10   bpp12 */ +	{ +		 54000000, { 0x091c, 0x091c, 0x06dc }, +	}, { +		 58400000, { 0x091c, 0x06dc, 0x06dc }, +	}, { +		 72000000, { 0x06dc, 0x06dc, 0x091c }, +	}, { +		 74250000, { 0x06dc, 0x0b5c, 0x091c }, +	}, { +		118800000, { 0x091c, 0x091c, 0x06dc }, +	}, { +		216000000, { 0x06dc, 0x0b5c, 0x091c }, +	} +}; + +static int hdmi_phy_configure(struct imx_hdmi *hdmi, unsigned char prep, +			      unsigned char res, int cscon) +{ +	unsigned res_idx, i; +	u8 val, msec; + +	if (prep) +		return -EINVAL; + +	switch (res) { +	case 0:	/* color resolution 0 is 8 bit colour depth */ +	case 8: +		res_idx = RES_8; +		break; +	case 10: +		res_idx = RES_10; +		break; +	case 12: +		res_idx = RES_12; +		break; +	default: +		return -EINVAL; +	} + +	/* Enable csc path */ +	if (cscon) +		val = HDMI_MC_FLOWCTRL_FEED_THROUGH_OFF_CSC_IN_PATH; +	else +		val = HDMI_MC_FLOWCTRL_FEED_THROUGH_OFF_CSC_BYPASS; + +	hdmi_writeb(hdmi, val, HDMI_MC_FLOWCTRL); + +	/* gen2 tx power off */ +	imx_hdmi_phy_gen2_txpwron(hdmi, 0); + +	/* gen2 pddq */ +	imx_hdmi_phy_gen2_pddq(hdmi, 1); + +	/* PHY reset */ +	hdmi_writeb(hdmi, HDMI_MC_PHYRSTZ_DEASSERT, HDMI_MC_PHYRSTZ); +	hdmi_writeb(hdmi, HDMI_MC_PHYRSTZ_ASSERT, HDMI_MC_PHYRSTZ); + +	hdmi_writeb(hdmi, HDMI_MC_HEACPHY_RST_ASSERT, HDMI_MC_HEACPHY_RST); + +	hdmi_phy_test_clear(hdmi, 1); +	hdmi_writeb(hdmi, HDMI_PHY_I2CM_SLAVE_ADDR_PHY_GEN2, +			HDMI_PHY_I2CM_SLAVE_ADDR); +	hdmi_phy_test_clear(hdmi, 0); + +	/* PLL/MPLL Cfg - always match on final entry */ +	for (i = 0; i < ARRAY_SIZE(mpll_config) - 1; i++) +		if (hdmi->hdmi_data.video_mode.mpixelclock <= +		    mpll_config[i].mpixelclock) +			break; + +	hdmi_phy_i2c_write(hdmi, mpll_config[i].res[res_idx].cpce, 0x06); +	hdmi_phy_i2c_write(hdmi, mpll_config[i].res[res_idx].gmp, 0x15); + +	for (i = 0; i < ARRAY_SIZE(curr_ctrl); i++) +		if (hdmi->hdmi_data.video_mode.mpixelclock <= +		    curr_ctrl[i].mpixelclock) +			break; + +	if (i >= ARRAY_SIZE(curr_ctrl)) { +		dev_err(hdmi->dev, +				"Pixel clock %d - unsupported by HDMI\n", +				hdmi->hdmi_data.video_mode.mpixelclock); +		return -EINVAL; +	} + +	/* CURRCTRL */ +	hdmi_phy_i2c_write(hdmi, curr_ctrl[i].curr[res_idx], 0x10); + +	hdmi_phy_i2c_write(hdmi, 0x0000, 0x13);  /* PLLPHBYCTRL */ +	hdmi_phy_i2c_write(hdmi, 0x0006, 0x17); +	/* RESISTANCE TERM 133Ohm Cfg */ +	hdmi_phy_i2c_write(hdmi, 0x0005, 0x19);  /* TXTERM */ +	/* PREEMP Cgf 0.00 */ +	hdmi_phy_i2c_write(hdmi, 0x800d, 0x09);  /* CKSYMTXCTRL */ +	/* TX/CK LVL 10 */ +	hdmi_phy_i2c_write(hdmi, 0x01ad, 0x0E);  /* VLEVCTRL */ +	/* REMOVE CLK TERM */ +	hdmi_phy_i2c_write(hdmi, 0x8000, 0x05);  /* CKCALCTRL */ + +	imx_hdmi_phy_enable_power(hdmi, 1); + +	/* toggle TMDS enable */ +	imx_hdmi_phy_enable_tmds(hdmi, 0); +	imx_hdmi_phy_enable_tmds(hdmi, 1); + +	/* gen2 tx power on */ +	imx_hdmi_phy_gen2_txpwron(hdmi, 1); +	imx_hdmi_phy_gen2_pddq(hdmi, 0); + +	/*Wait for PHY PLL lock */ +	msec = 5; +	do { +		val = hdmi_readb(hdmi, HDMI_PHY_STAT0) & HDMI_PHY_TX_PHY_LOCK; +		if (!val) +			break; + +		if (msec == 0) { +			dev_err(hdmi->dev, "PHY PLL not locked\n"); +			return -ETIMEDOUT; +		} + +		udelay(1000); +		msec--; +	} while (1); + +	return 0; +} + +static int imx_hdmi_phy_init(struct imx_hdmi *hdmi) +{ +	int i, ret; +	bool cscon = false; + +	/*check csc whether needed activated in HDMI mode */ +	cscon = (is_color_space_conversion(hdmi) && +			!hdmi->hdmi_data.video_mode.mdvi); + +	/* HDMI Phy spec says to do the phy initialization sequence twice */ +	for (i = 0; i < 2; i++) { +		imx_hdmi_phy_sel_data_en_pol(hdmi, 1); +		imx_hdmi_phy_sel_interface_control(hdmi, 0); +		imx_hdmi_phy_enable_tmds(hdmi, 0); +		imx_hdmi_phy_enable_power(hdmi, 0); + +		/* Enable CSC */ +		ret = hdmi_phy_configure(hdmi, 0, 8, cscon); +		if (ret) +			return ret; +	} + +	hdmi->phy_enabled = true; +	return 0; +} + +static void hdmi_tx_hdcp_config(struct imx_hdmi *hdmi) +{ +	u8 de; + +	if (hdmi->hdmi_data.video_mode.mdataenablepolarity) +		de = HDMI_A_VIDPOLCFG_DATAENPOL_ACTIVE_HIGH; +	else +		de = HDMI_A_VIDPOLCFG_DATAENPOL_ACTIVE_LOW; + +	/* disable rx detect */ +	hdmi_modb(hdmi, HDMI_A_HDCPCFG0_RXDETECT_DISABLE, +		  HDMI_A_HDCPCFG0_RXDETECT_MASK, HDMI_A_HDCPCFG0); + +	hdmi_modb(hdmi, de, HDMI_A_VIDPOLCFG_DATAENPOL_MASK, HDMI_A_VIDPOLCFG); + +	hdmi_modb(hdmi, HDMI_A_HDCPCFG1_ENCRYPTIONDISABLE_DISABLE, +		  HDMI_A_HDCPCFG1_ENCRYPTIONDISABLE_MASK, HDMI_A_HDCPCFG1); +} + +static void hdmi_config_AVI(struct imx_hdmi *hdmi) +{ +	u8 val, pix_fmt, under_scan; +	u8 act_ratio, coded_ratio, colorimetry, ext_colorimetry; +	bool aspect_16_9; + +	aspect_16_9 = false; /* FIXME */ + +	/* AVI Data Byte 1 */ +	if (hdmi->hdmi_data.enc_out_format == YCBCR444) +		pix_fmt = HDMI_FC_AVICONF0_PIX_FMT_YCBCR444; +	else if (hdmi->hdmi_data.enc_out_format == YCBCR422_8BITS) +		pix_fmt = HDMI_FC_AVICONF0_PIX_FMT_YCBCR422; +	else +		pix_fmt = HDMI_FC_AVICONF0_PIX_FMT_RGB; + +		under_scan =  HDMI_FC_AVICONF0_SCAN_INFO_NODATA; + +	/* +	 * Active format identification data is present in the AVI InfoFrame. +	 * Under scan info, no bar data +	 */ +	val = pix_fmt | under_scan | +		HDMI_FC_AVICONF0_ACTIVE_FMT_INFO_PRESENT | +		HDMI_FC_AVICONF0_BAR_DATA_NO_DATA; + +	hdmi_writeb(hdmi, val, HDMI_FC_AVICONF0); + +	/* AVI Data Byte 2 -Set the Aspect Ratio */ +	if (aspect_16_9) { +		act_ratio = HDMI_FC_AVICONF1_ACTIVE_ASPECT_RATIO_16_9; +		coded_ratio = HDMI_FC_AVICONF1_CODED_ASPECT_RATIO_16_9; +	} else { +		act_ratio = HDMI_FC_AVICONF1_ACTIVE_ASPECT_RATIO_4_3; +		coded_ratio = HDMI_FC_AVICONF1_CODED_ASPECT_RATIO_4_3; +	} + +	/* Set up colorimetry */ +	if (hdmi->hdmi_data.enc_out_format == XVYCC444) { +		colorimetry = HDMI_FC_AVICONF1_COLORIMETRY_EXTENDED_INFO; +		if (hdmi->hdmi_data.colorimetry == HDMI_COLORIMETRY_ITU_601) +			ext_colorimetry = +				HDMI_FC_AVICONF2_EXT_COLORIMETRY_XVYCC601; +		else /*hdmi->hdmi_data.colorimetry == HDMI_COLORIMETRY_ITU_709*/ +			ext_colorimetry = +				HDMI_FC_AVICONF2_EXT_COLORIMETRY_XVYCC709; +	} else if (hdmi->hdmi_data.enc_out_format != RGB) { +		if (hdmi->hdmi_data.colorimetry == HDMI_COLORIMETRY_ITU_601) +			colorimetry = HDMI_FC_AVICONF1_COLORIMETRY_SMPTE; +		else /*hdmi->hdmi_data.colorimetry == HDMI_COLORIMETRY_ITU_709*/ +			colorimetry = HDMI_FC_AVICONF1_COLORIMETRY_ITUR; +		ext_colorimetry = HDMI_FC_AVICONF2_EXT_COLORIMETRY_XVYCC601; +	} else { /* Carries no data */ +		colorimetry = HDMI_FC_AVICONF1_COLORIMETRY_NO_DATA; +		ext_colorimetry = HDMI_FC_AVICONF2_EXT_COLORIMETRY_XVYCC601; +	} + +	val = colorimetry | coded_ratio | act_ratio; +	hdmi_writeb(hdmi, val, HDMI_FC_AVICONF1); + +	/* AVI Data Byte 3 */ +	val = HDMI_FC_AVICONF2_IT_CONTENT_NO_DATA | ext_colorimetry | +		HDMI_FC_AVICONF2_RGB_QUANT_DEFAULT | +		HDMI_FC_AVICONF2_SCALING_NONE; +	hdmi_writeb(hdmi, val, HDMI_FC_AVICONF2); + +	/* AVI Data Byte 4 */ +	hdmi_writeb(hdmi, hdmi->vic, HDMI_FC_AVIVID); + +	/* AVI Data Byte 5- set up input and output pixel repetition */ +	val = (((hdmi->hdmi_data.video_mode.mpixelrepetitioninput + 1) << +		HDMI_FC_PRCONF_INCOMING_PR_FACTOR_OFFSET) & +		HDMI_FC_PRCONF_INCOMING_PR_FACTOR_MASK) | +		((hdmi->hdmi_data.video_mode.mpixelrepetitionoutput << +		HDMI_FC_PRCONF_OUTPUT_PR_FACTOR_OFFSET) & +		HDMI_FC_PRCONF_OUTPUT_PR_FACTOR_MASK); +	hdmi_writeb(hdmi, val, HDMI_FC_PRCONF); + +	/* IT Content and quantization range = don't care */ +	val = HDMI_FC_AVICONF3_IT_CONTENT_TYPE_GRAPHICS | +		HDMI_FC_AVICONF3_QUANT_RANGE_LIMITED; +	hdmi_writeb(hdmi, val, HDMI_FC_AVICONF3); + +	/* AVI Data Bytes 6-13 */ +	hdmi_writeb(hdmi, 0, HDMI_FC_AVIETB0); +	hdmi_writeb(hdmi, 0, HDMI_FC_AVIETB1); +	hdmi_writeb(hdmi, 0, HDMI_FC_AVISBB0); +	hdmi_writeb(hdmi, 0, HDMI_FC_AVISBB1); +	hdmi_writeb(hdmi, 0, HDMI_FC_AVIELB0); +	hdmi_writeb(hdmi, 0, HDMI_FC_AVIELB1); +	hdmi_writeb(hdmi, 0, HDMI_FC_AVISRB0); +	hdmi_writeb(hdmi, 0, HDMI_FC_AVISRB1); +} + +static void hdmi_av_composer(struct imx_hdmi *hdmi, +			     const struct drm_display_mode *mode) +{ +	u8 inv_val; +	struct hdmi_vmode *vmode = &hdmi->hdmi_data.video_mode; +	int hblank, vblank, h_de_hs, v_de_vs, hsync_len, vsync_len; + +	vmode->mhsyncpolarity = !!(mode->flags & DRM_MODE_FLAG_PHSYNC); +	vmode->mvsyncpolarity = !!(mode->flags & DRM_MODE_FLAG_PVSYNC); +	vmode->minterlaced = !!(mode->flags & DRM_MODE_FLAG_INTERLACE); +	vmode->mpixelclock = mode->clock * 1000; + +	dev_dbg(hdmi->dev, "final pixclk = %d\n", vmode->mpixelclock); + +	/* Set up HDMI_FC_INVIDCONF */ +	inv_val = (hdmi->hdmi_data.hdcp_enable ? +		HDMI_FC_INVIDCONF_HDCP_KEEPOUT_ACTIVE : +		HDMI_FC_INVIDCONF_HDCP_KEEPOUT_INACTIVE); + +	inv_val |= (vmode->mvsyncpolarity ? +		HDMI_FC_INVIDCONF_VSYNC_IN_POLARITY_ACTIVE_HIGH : +		HDMI_FC_INVIDCONF_VSYNC_IN_POLARITY_ACTIVE_LOW); + +	inv_val |= (vmode->mhsyncpolarity ? +		HDMI_FC_INVIDCONF_HSYNC_IN_POLARITY_ACTIVE_HIGH : +		HDMI_FC_INVIDCONF_HSYNC_IN_POLARITY_ACTIVE_LOW); + +	inv_val |= (vmode->mdataenablepolarity ? +		HDMI_FC_INVIDCONF_DE_IN_POLARITY_ACTIVE_HIGH : +		HDMI_FC_INVIDCONF_DE_IN_POLARITY_ACTIVE_LOW); + +	if (hdmi->vic == 39) +		inv_val |= HDMI_FC_INVIDCONF_R_V_BLANK_IN_OSC_ACTIVE_HIGH; +	else +		inv_val |= (vmode->minterlaced ? +			HDMI_FC_INVIDCONF_R_V_BLANK_IN_OSC_ACTIVE_HIGH : +			HDMI_FC_INVIDCONF_R_V_BLANK_IN_OSC_ACTIVE_LOW); + +	inv_val |= (vmode->minterlaced ? +		HDMI_FC_INVIDCONF_IN_I_P_INTERLACED : +		HDMI_FC_INVIDCONF_IN_I_P_PROGRESSIVE); + +	inv_val |= (vmode->mdvi ? +		HDMI_FC_INVIDCONF_DVI_MODEZ_DVI_MODE : +		HDMI_FC_INVIDCONF_DVI_MODEZ_HDMI_MODE); + +	hdmi_writeb(hdmi, inv_val, HDMI_FC_INVIDCONF); + +	/* Set up horizontal active pixel width */ +	hdmi_writeb(hdmi, mode->hdisplay >> 8, HDMI_FC_INHACTV1); +	hdmi_writeb(hdmi, mode->hdisplay, HDMI_FC_INHACTV0); + +	/* Set up vertical active lines */ +	hdmi_writeb(hdmi, mode->vdisplay >> 8, HDMI_FC_INVACTV1); +	hdmi_writeb(hdmi, mode->vdisplay, HDMI_FC_INVACTV0); + +	/* Set up horizontal blanking pixel region width */ +	hblank = mode->htotal - mode->hdisplay; +	hdmi_writeb(hdmi, hblank >> 8, HDMI_FC_INHBLANK1); +	hdmi_writeb(hdmi, hblank, HDMI_FC_INHBLANK0); + +	/* Set up vertical blanking pixel region width */ +	vblank = mode->vtotal - mode->vdisplay; +	hdmi_writeb(hdmi, vblank, HDMI_FC_INVBLANK); + +	/* Set up HSYNC active edge delay width (in pixel clks) */ +	h_de_hs = mode->hsync_start - mode->hdisplay; +	hdmi_writeb(hdmi, h_de_hs >> 8, HDMI_FC_HSYNCINDELAY1); +	hdmi_writeb(hdmi, h_de_hs, HDMI_FC_HSYNCINDELAY0); + +	/* Set up VSYNC active edge delay (in lines) */ +	v_de_vs = mode->vsync_start - mode->vdisplay; +	hdmi_writeb(hdmi, v_de_vs, HDMI_FC_VSYNCINDELAY); + +	/* Set up HSYNC active pulse width (in pixel clks) */ +	hsync_len = mode->hsync_end - mode->hsync_start; +	hdmi_writeb(hdmi, hsync_len >> 8, HDMI_FC_HSYNCINWIDTH1); +	hdmi_writeb(hdmi, hsync_len, HDMI_FC_HSYNCINWIDTH0); + +	/* Set up VSYNC active edge delay (in lines) */ +	vsync_len = mode->vsync_end - mode->vsync_start; +	hdmi_writeb(hdmi, vsync_len, HDMI_FC_VSYNCINWIDTH); +} + +static void imx_hdmi_phy_disable(struct imx_hdmi *hdmi) +{ +	if (!hdmi->phy_enabled) +		return; + +	imx_hdmi_phy_enable_tmds(hdmi, 0); +	imx_hdmi_phy_enable_power(hdmi, 0); + +	hdmi->phy_enabled = false; +} + +/* HDMI Initialization Step B.4 */ +static void imx_hdmi_enable_video_path(struct imx_hdmi *hdmi) +{ +	u8 clkdis; + +	/* control period minimum duration */ +	hdmi_writeb(hdmi, 12, HDMI_FC_CTRLDUR); +	hdmi_writeb(hdmi, 32, HDMI_FC_EXCTRLDUR); +	hdmi_writeb(hdmi, 1, HDMI_FC_EXCTRLSPAC); + +	/* Set to fill TMDS data channels */ +	hdmi_writeb(hdmi, 0x0B, HDMI_FC_CH0PREAM); +	hdmi_writeb(hdmi, 0x16, HDMI_FC_CH1PREAM); +	hdmi_writeb(hdmi, 0x21, HDMI_FC_CH2PREAM); + +	/* Enable pixel clock and tmds data path */ +	clkdis = 0x7F; +	clkdis &= ~HDMI_MC_CLKDIS_PIXELCLK_DISABLE; +	hdmi_writeb(hdmi, clkdis, HDMI_MC_CLKDIS); + +	clkdis &= ~HDMI_MC_CLKDIS_TMDSCLK_DISABLE; +	hdmi_writeb(hdmi, clkdis, HDMI_MC_CLKDIS); + +	/* Enable csc path */ +	if (is_color_space_conversion(hdmi)) { +		clkdis &= ~HDMI_MC_CLKDIS_CSCCLK_DISABLE; +		hdmi_writeb(hdmi, clkdis, HDMI_MC_CLKDIS); +	} +} + +static void hdmi_enable_audio_clk(struct imx_hdmi *hdmi) +{ +	hdmi_modb(hdmi, 0, HDMI_MC_CLKDIS_AUDCLK_DISABLE, HDMI_MC_CLKDIS); +} + +/* Workaround to clear the overflow condition */ +static void imx_hdmi_clear_overflow(struct imx_hdmi *hdmi) +{ +	int count; +	u8 val; + +	/* TMDS software reset */ +	hdmi_writeb(hdmi, (u8)~HDMI_MC_SWRSTZ_TMDSSWRST_REQ, HDMI_MC_SWRSTZ); + +	val = hdmi_readb(hdmi, HDMI_FC_INVIDCONF); +	if (hdmi->dev_type == IMX6DL_HDMI) { +		hdmi_writeb(hdmi, val, HDMI_FC_INVIDCONF); +		return; +	} + +	for (count = 0; count < 4; count++) +		hdmi_writeb(hdmi, val, HDMI_FC_INVIDCONF); +} + +static void hdmi_enable_overflow_interrupts(struct imx_hdmi *hdmi) +{ +	hdmi_writeb(hdmi, 0, HDMI_FC_MASK2); +	hdmi_writeb(hdmi, 0, HDMI_IH_MUTE_FC_STAT2); +} + +static void hdmi_disable_overflow_interrupts(struct imx_hdmi *hdmi) +{ +	hdmi_writeb(hdmi, HDMI_IH_MUTE_FC_STAT2_OVERFLOW_MASK, +		    HDMI_IH_MUTE_FC_STAT2); +} + +static int imx_hdmi_setup(struct imx_hdmi *hdmi, struct drm_display_mode *mode) +{ +	int ret; + +	hdmi_disable_overflow_interrupts(hdmi); + +	hdmi->vic = drm_match_cea_mode(mode); + +	if (!hdmi->vic) { +		dev_dbg(hdmi->dev, "Non-CEA mode used in HDMI\n"); +		hdmi->hdmi_data.video_mode.mdvi = true; +	} else { +		dev_dbg(hdmi->dev, "CEA mode used vic=%d\n", hdmi->vic); +		hdmi->hdmi_data.video_mode.mdvi = false; +	} + +	if ((hdmi->vic == 6) || (hdmi->vic == 7) || +		(hdmi->vic == 21) || (hdmi->vic == 22) || +		(hdmi->vic == 2) || (hdmi->vic == 3) || +		(hdmi->vic == 17) || (hdmi->vic == 18)) +		hdmi->hdmi_data.colorimetry = HDMI_COLORIMETRY_ITU_601; +	else +		hdmi->hdmi_data.colorimetry = HDMI_COLORIMETRY_ITU_709; + +	if ((hdmi->vic == 10) || (hdmi->vic == 11) || +		(hdmi->vic == 12) || (hdmi->vic == 13) || +		(hdmi->vic == 14) || (hdmi->vic == 15) || +		(hdmi->vic == 25) || (hdmi->vic == 26) || +		(hdmi->vic == 27) || (hdmi->vic == 28) || +		(hdmi->vic == 29) || (hdmi->vic == 30) || +		(hdmi->vic == 35) || (hdmi->vic == 36) || +		(hdmi->vic == 37) || (hdmi->vic == 38)) +		hdmi->hdmi_data.video_mode.mpixelrepetitionoutput = 1; +	else +		hdmi->hdmi_data.video_mode.mpixelrepetitionoutput = 0; + +	hdmi->hdmi_data.video_mode.mpixelrepetitioninput = 0; + +	/* TODO: Get input format from IPU (via FB driver interface) */ +	hdmi->hdmi_data.enc_in_format = RGB; + +	hdmi->hdmi_data.enc_out_format = RGB; + +	hdmi->hdmi_data.enc_color_depth = 8; +	hdmi->hdmi_data.pix_repet_factor = 0; +	hdmi->hdmi_data.hdcp_enable = 0; +	hdmi->hdmi_data.video_mode.mdataenablepolarity = true; + +	/* HDMI Initialization Step B.1 */ +	hdmi_av_composer(hdmi, mode); + +	/* HDMI Initializateion Step B.2 */ +	ret = imx_hdmi_phy_init(hdmi); +	if (ret) +		return ret; + +	/* HDMI Initialization Step B.3 */ +	imx_hdmi_enable_video_path(hdmi); + +	/* not for DVI mode */ +	if (hdmi->hdmi_data.video_mode.mdvi) +		dev_dbg(hdmi->dev, "%s DVI mode\n", __func__); +	else { +		dev_dbg(hdmi->dev, "%s CEA mode\n", __func__); + +		/* HDMI Initialization Step E - Configure audio */ +		hdmi_clk_regenerator_update_pixel_clock(hdmi); +		hdmi_enable_audio_clk(hdmi); + +		/* HDMI Initialization Step F - Configure AVI InfoFrame */ +		hdmi_config_AVI(hdmi); +	} + +	hdmi_video_packetize(hdmi); +	hdmi_video_csc(hdmi); +	hdmi_video_sample(hdmi); +	hdmi_tx_hdcp_config(hdmi); + +	imx_hdmi_clear_overflow(hdmi); +	if (hdmi->cable_plugin && !hdmi->hdmi_data.video_mode.mdvi) +		hdmi_enable_overflow_interrupts(hdmi); + +	return 0; +} + +/* Wait until we are registered to enable interrupts */ +static int imx_hdmi_fb_registered(struct imx_hdmi *hdmi) +{ +	hdmi_writeb(hdmi, HDMI_PHY_I2CM_INT_ADDR_DONE_POL, +		    HDMI_PHY_I2CM_INT_ADDR); + +	hdmi_writeb(hdmi, HDMI_PHY_I2CM_CTLINT_ADDR_NAC_POL | +		    HDMI_PHY_I2CM_CTLINT_ADDR_ARBITRATION_POL, +		    HDMI_PHY_I2CM_CTLINT_ADDR); + +	/* enable cable hot plug irq */ +	hdmi_writeb(hdmi, (u8)~HDMI_PHY_HPD, HDMI_PHY_MASK0); + +	/* Clear Hotplug interrupts */ +	hdmi_writeb(hdmi, HDMI_IH_PHY_STAT0_HPD, HDMI_IH_PHY_STAT0); + +	return 0; +} + +static void initialize_hdmi_ih_mutes(struct imx_hdmi *hdmi) +{ +	u8 ih_mute; + +	/* +	 * Boot up defaults are: +	 * HDMI_IH_MUTE   = 0x03 (disabled) +	 * HDMI_IH_MUTE_* = 0x00 (enabled) +	 * +	 * Disable top level interrupt bits in HDMI block +	 */ +	ih_mute = hdmi_readb(hdmi, HDMI_IH_MUTE) | +		  HDMI_IH_MUTE_MUTE_WAKEUP_INTERRUPT | +		  HDMI_IH_MUTE_MUTE_ALL_INTERRUPT; + +	hdmi_writeb(hdmi, ih_mute, HDMI_IH_MUTE); + +	/* by default mask all interrupts */ +	hdmi_writeb(hdmi, 0xff, HDMI_VP_MASK); +	hdmi_writeb(hdmi, 0xff, HDMI_FC_MASK0); +	hdmi_writeb(hdmi, 0xff, HDMI_FC_MASK1); +	hdmi_writeb(hdmi, 0xff, HDMI_FC_MASK2); +	hdmi_writeb(hdmi, 0xff, HDMI_PHY_MASK0); +	hdmi_writeb(hdmi, 0xff, HDMI_PHY_I2CM_INT_ADDR); +	hdmi_writeb(hdmi, 0xff, HDMI_PHY_I2CM_CTLINT_ADDR); +	hdmi_writeb(hdmi, 0xff, HDMI_AUD_INT); +	hdmi_writeb(hdmi, 0xff, HDMI_AUD_SPDIFINT); +	hdmi_writeb(hdmi, 0xff, HDMI_AUD_HBR_MASK); +	hdmi_writeb(hdmi, 0xff, HDMI_GP_MASK); +	hdmi_writeb(hdmi, 0xff, HDMI_A_APIINTMSK); +	hdmi_writeb(hdmi, 0xff, HDMI_CEC_MASK); +	hdmi_writeb(hdmi, 0xff, HDMI_I2CM_INT); +	hdmi_writeb(hdmi, 0xff, HDMI_I2CM_CTLINT); + +	/* Disable interrupts in the IH_MUTE_* registers */ +	hdmi_writeb(hdmi, 0xff, HDMI_IH_MUTE_FC_STAT0); +	hdmi_writeb(hdmi, 0xff, HDMI_IH_MUTE_FC_STAT1); +	hdmi_writeb(hdmi, 0xff, HDMI_IH_MUTE_FC_STAT2); +	hdmi_writeb(hdmi, 0xff, HDMI_IH_MUTE_AS_STAT0); +	hdmi_writeb(hdmi, 0xff, HDMI_IH_MUTE_PHY_STAT0); +	hdmi_writeb(hdmi, 0xff, HDMI_IH_MUTE_I2CM_STAT0); +	hdmi_writeb(hdmi, 0xff, HDMI_IH_MUTE_CEC_STAT0); +	hdmi_writeb(hdmi, 0xff, HDMI_IH_MUTE_VP_STAT0); +	hdmi_writeb(hdmi, 0xff, HDMI_IH_MUTE_I2CMPHY_STAT0); +	hdmi_writeb(hdmi, 0xff, HDMI_IH_MUTE_AHBDMAAUD_STAT0); + +	/* Enable top level interrupt bits in HDMI block */ +	ih_mute &= ~(HDMI_IH_MUTE_MUTE_WAKEUP_INTERRUPT | +		    HDMI_IH_MUTE_MUTE_ALL_INTERRUPT); +	hdmi_writeb(hdmi, ih_mute, HDMI_IH_MUTE); +} + +static void imx_hdmi_poweron(struct imx_hdmi *hdmi) +{ +	imx_hdmi_setup(hdmi, &hdmi->previous_mode); +} + +static void imx_hdmi_poweroff(struct imx_hdmi *hdmi) +{ +	imx_hdmi_phy_disable(hdmi); +} + +static enum drm_connector_status imx_hdmi_connector_detect(struct drm_connector +							*connector, bool force) +{ +	struct imx_hdmi *hdmi = container_of(connector, struct imx_hdmi, +					     connector); + +	return hdmi_readb(hdmi, HDMI_PHY_STAT0) & HDMI_PHY_HPD ? +		connector_status_connected : connector_status_disconnected; +} + +static int imx_hdmi_connector_get_modes(struct drm_connector *connector) +{ +	struct imx_hdmi *hdmi = container_of(connector, struct imx_hdmi, +					     connector); +	struct edid *edid; +	int ret; + +	if (!hdmi->ddc) +		return 0; + +	edid = drm_get_edid(connector, hdmi->ddc); +	if (edid) { +		dev_dbg(hdmi->dev, "got edid: width[%d] x height[%d]\n", +			edid->width_cm, edid->height_cm); + +		drm_mode_connector_update_edid_property(connector, edid); +		ret = drm_add_edid_modes(connector, edid); +		kfree(edid); +	} else { +		dev_dbg(hdmi->dev, "failed to get edid\n"); +	} + +	return 0; +} + +static struct drm_encoder *imx_hdmi_connector_best_encoder(struct drm_connector +							   *connector) +{ +	struct imx_hdmi *hdmi = container_of(connector, struct imx_hdmi, +					     connector); + +	return &hdmi->encoder; +} + +static void imx_hdmi_encoder_mode_set(struct drm_encoder *encoder, +			struct drm_display_mode *mode, +			struct drm_display_mode *adjusted_mode) +{ +	struct imx_hdmi *hdmi = container_of(encoder, struct imx_hdmi, encoder); + +	imx_hdmi_setup(hdmi, mode); + +	/* Store the display mode for plugin/DKMS poweron events */ +	memcpy(&hdmi->previous_mode, mode, sizeof(hdmi->previous_mode)); +} + +static bool imx_hdmi_encoder_mode_fixup(struct drm_encoder *encoder, +			const struct drm_display_mode *mode, +			struct drm_display_mode *adjusted_mode) +{ +	return true; +} + +static void imx_hdmi_encoder_disable(struct drm_encoder *encoder) +{ +} + +static void imx_hdmi_encoder_dpms(struct drm_encoder *encoder, int mode) +{ +	struct imx_hdmi *hdmi = container_of(encoder, struct imx_hdmi, encoder); + +	if (mode) +		imx_hdmi_poweroff(hdmi); +	else +		imx_hdmi_poweron(hdmi); +} + +static void imx_hdmi_encoder_prepare(struct drm_encoder *encoder) +{ +	struct imx_hdmi *hdmi = container_of(encoder, struct imx_hdmi, encoder); + +	imx_hdmi_poweroff(hdmi); +	imx_drm_panel_format(encoder, V4L2_PIX_FMT_RGB24); +} + +static void imx_hdmi_encoder_commit(struct drm_encoder *encoder) +{ +	struct imx_hdmi *hdmi = container_of(encoder, struct imx_hdmi, encoder); +	int mux = imx_drm_encoder_get_mux_id(hdmi->dev->of_node, encoder); + +	imx_hdmi_set_ipu_di_mux(hdmi, mux); + +	imx_hdmi_poweron(hdmi); +} + +static struct drm_encoder_funcs imx_hdmi_encoder_funcs = { +	.destroy = imx_drm_encoder_destroy, +}; + +static struct drm_encoder_helper_funcs imx_hdmi_encoder_helper_funcs = { +	.dpms = imx_hdmi_encoder_dpms, +	.prepare = imx_hdmi_encoder_prepare, +	.commit = imx_hdmi_encoder_commit, +	.mode_set = imx_hdmi_encoder_mode_set, +	.mode_fixup = imx_hdmi_encoder_mode_fixup, +	.disable = imx_hdmi_encoder_disable, +}; + +static struct drm_connector_funcs imx_hdmi_connector_funcs = { +	.dpms = drm_helper_connector_dpms, +	.fill_modes = drm_helper_probe_single_connector_modes, +	.detect = imx_hdmi_connector_detect, +	.destroy = imx_drm_connector_destroy, +}; + +static struct drm_connector_helper_funcs imx_hdmi_connector_helper_funcs = { +	.get_modes = imx_hdmi_connector_get_modes, +	.best_encoder = imx_hdmi_connector_best_encoder, +}; + +static irqreturn_t imx_hdmi_hardirq(int irq, void *dev_id) +{ +	struct imx_hdmi *hdmi = dev_id; +	u8 intr_stat; + +	intr_stat = hdmi_readb(hdmi, HDMI_IH_PHY_STAT0); +	if (intr_stat) +		hdmi_writeb(hdmi, ~0, HDMI_IH_MUTE_PHY_STAT0); + +	return intr_stat ? IRQ_WAKE_THREAD : IRQ_NONE; +} + +static irqreturn_t imx_hdmi_irq(int irq, void *dev_id) +{ +	struct imx_hdmi *hdmi = dev_id; +	u8 intr_stat; +	u8 phy_int_pol; + +	intr_stat = hdmi_readb(hdmi, HDMI_IH_PHY_STAT0); + +	phy_int_pol = hdmi_readb(hdmi, HDMI_PHY_POL0); + +	if (intr_stat & HDMI_IH_PHY_STAT0_HPD) { +		if (phy_int_pol & HDMI_PHY_HPD) { +			dev_dbg(hdmi->dev, "EVENT=plugin\n"); + +			hdmi_modb(hdmi, 0, HDMI_PHY_HPD, HDMI_PHY_POL0); + +			imx_hdmi_poweron(hdmi); +		} else { +			dev_dbg(hdmi->dev, "EVENT=plugout\n"); + +			hdmi_modb(hdmi, HDMI_PHY_HPD, HDMI_PHY_HPD, +				HDMI_PHY_POL0); + +			imx_hdmi_poweroff(hdmi); +		} +		drm_helper_hpd_irq_event(hdmi->connector.dev); +	} + +	hdmi_writeb(hdmi, intr_stat, HDMI_IH_PHY_STAT0); +	hdmi_writeb(hdmi, ~HDMI_IH_PHY_STAT0_HPD, HDMI_IH_MUTE_PHY_STAT0); + +	return IRQ_HANDLED; +} + +static int imx_hdmi_register(struct drm_device *drm, struct imx_hdmi *hdmi) +{ +	int ret; + +	ret = imx_drm_encoder_parse_of(drm, &hdmi->encoder, +				       hdmi->dev->of_node); +	if (ret) +		return ret; + +	hdmi->connector.polled = DRM_CONNECTOR_POLL_HPD; + +	drm_encoder_helper_add(&hdmi->encoder, &imx_hdmi_encoder_helper_funcs); +	drm_encoder_init(drm, &hdmi->encoder, &imx_hdmi_encoder_funcs, +			 DRM_MODE_ENCODER_TMDS); + +	drm_connector_helper_add(&hdmi->connector, +			&imx_hdmi_connector_helper_funcs); +	drm_connector_init(drm, &hdmi->connector, &imx_hdmi_connector_funcs, +			   DRM_MODE_CONNECTOR_HDMIA); + +	hdmi->connector.encoder = &hdmi->encoder; + +	drm_mode_connector_attach_encoder(&hdmi->connector, &hdmi->encoder); + +	return 0; +} + +static struct platform_device_id imx_hdmi_devtype[] = { +	{ +		.name = "imx6q-hdmi", +		.driver_data = IMX6Q_HDMI, +	}, { +		.name = "imx6dl-hdmi", +		.driver_data = IMX6DL_HDMI, +	}, { /* sentinel */ } +}; +MODULE_DEVICE_TABLE(platform, imx_hdmi_devtype); + +static const struct of_device_id imx_hdmi_dt_ids[] = { +{ .compatible = "fsl,imx6q-hdmi", .data = &imx_hdmi_devtype[IMX6Q_HDMI], }, +{ .compatible = "fsl,imx6dl-hdmi", .data = &imx_hdmi_devtype[IMX6DL_HDMI], }, +{ /* sentinel */ } +}; +MODULE_DEVICE_TABLE(of, imx_hdmi_dt_ids); + +static int imx_hdmi_bind(struct device *dev, struct device *master, void *data) +{ +	struct platform_device *pdev = to_platform_device(dev); +	const struct of_device_id *of_id = +				of_match_device(imx_hdmi_dt_ids, dev); +	struct drm_device *drm = data; +	struct device_node *np = dev->of_node; +	struct device_node *ddc_node; +	struct imx_hdmi *hdmi; +	struct resource *iores; +	int ret, irq; + +	hdmi = devm_kzalloc(dev, sizeof(*hdmi), GFP_KERNEL); +	if (!hdmi) +		return -ENOMEM; + +	hdmi->dev = dev; +	hdmi->sample_rate = 48000; +	hdmi->ratio = 100; + +	if (of_id) { +		const struct platform_device_id *device_id = of_id->data; + +		hdmi->dev_type = device_id->driver_data; +	} + +	ddc_node = of_parse_phandle(np, "ddc-i2c-bus", 0); +	if (ddc_node) { +		hdmi->ddc = of_find_i2c_adapter_by_node(ddc_node); +		if (!hdmi->ddc) +			dev_dbg(hdmi->dev, "failed to read ddc node\n"); + +		of_node_put(ddc_node); +	} else { +		dev_dbg(hdmi->dev, "no ddc property found\n"); +	} + +	irq = platform_get_irq(pdev, 0); +	if (irq < 0) +		return irq; + +	ret = devm_request_threaded_irq(dev, irq, imx_hdmi_hardirq, +					imx_hdmi_irq, IRQF_SHARED, +					dev_name(dev), hdmi); +	if (ret) +		return ret; + +	iores = platform_get_resource(pdev, IORESOURCE_MEM, 0); +	hdmi->regs = devm_ioremap_resource(dev, iores); +	if (IS_ERR(hdmi->regs)) +		return PTR_ERR(hdmi->regs); + +	hdmi->regmap = syscon_regmap_lookup_by_phandle(np, "gpr"); +	if (IS_ERR(hdmi->regmap)) +		return PTR_ERR(hdmi->regmap); + +	hdmi->isfr_clk = devm_clk_get(hdmi->dev, "isfr"); +	if (IS_ERR(hdmi->isfr_clk)) { +		ret = PTR_ERR(hdmi->isfr_clk); +		dev_err(hdmi->dev, +			"Unable to get HDMI isfr clk: %d\n", ret); +		return ret; +	} + +	ret = clk_prepare_enable(hdmi->isfr_clk); +	if (ret) { +		dev_err(hdmi->dev, +			"Cannot enable HDMI isfr clock: %d\n", ret); +		return ret; +	} + +	hdmi->iahb_clk = devm_clk_get(hdmi->dev, "iahb"); +	if (IS_ERR(hdmi->iahb_clk)) { +		ret = PTR_ERR(hdmi->iahb_clk); +		dev_err(hdmi->dev, +			"Unable to get HDMI iahb clk: %d\n", ret); +		goto err_isfr; +	} + +	ret = clk_prepare_enable(hdmi->iahb_clk); +	if (ret) { +		dev_err(hdmi->dev, +			"Cannot enable HDMI iahb clock: %d\n", ret); +		goto err_isfr; +	} + +	/* Product and revision IDs */ +	dev_info(dev, +		"Detected HDMI controller 0x%x:0x%x:0x%x:0x%x\n", +		hdmi_readb(hdmi, HDMI_DESIGN_ID), +		hdmi_readb(hdmi, HDMI_REVISION_ID), +		hdmi_readb(hdmi, HDMI_PRODUCT_ID0), +		hdmi_readb(hdmi, HDMI_PRODUCT_ID1)); + +	initialize_hdmi_ih_mutes(hdmi); + +	/* +	 * To prevent overflows in HDMI_IH_FC_STAT2, set the clk regenerator +	 * N and cts values before enabling phy +	 */ +	hdmi_init_clk_regenerator(hdmi); + +	/* +	 * Configure registers related to HDMI interrupt +	 * generation before registering IRQ. +	 */ +	hdmi_writeb(hdmi, HDMI_PHY_HPD, HDMI_PHY_POL0); + +	/* Clear Hotplug interrupts */ +	hdmi_writeb(hdmi, HDMI_IH_PHY_STAT0_HPD, HDMI_IH_PHY_STAT0); + +	ret = imx_hdmi_fb_registered(hdmi); +	if (ret) +		goto err_iahb; + +	ret = imx_hdmi_register(drm, hdmi); +	if (ret) +		goto err_iahb; + +	/* Unmute interrupts */ +	hdmi_writeb(hdmi, ~HDMI_IH_PHY_STAT0_HPD, HDMI_IH_MUTE_PHY_STAT0); + +	dev_set_drvdata(dev, hdmi); + +	return 0; + +err_iahb: +	clk_disable_unprepare(hdmi->iahb_clk); +err_isfr: +	clk_disable_unprepare(hdmi->isfr_clk); + +	return ret; +} + +static void imx_hdmi_unbind(struct device *dev, struct device *master, +	void *data) +{ +	struct imx_hdmi *hdmi = dev_get_drvdata(dev); + +	/* Disable all interrupts */ +	hdmi_writeb(hdmi, ~0, HDMI_IH_MUTE_PHY_STAT0); + +	hdmi->connector.funcs->destroy(&hdmi->connector); +	hdmi->encoder.funcs->destroy(&hdmi->encoder); + +	clk_disable_unprepare(hdmi->iahb_clk); +	clk_disable_unprepare(hdmi->isfr_clk); +	i2c_put_adapter(hdmi->ddc); +} + +static const struct component_ops hdmi_ops = { +	.bind	= imx_hdmi_bind, +	.unbind	= imx_hdmi_unbind, +}; + +static int imx_hdmi_platform_probe(struct platform_device *pdev) +{ +	return component_add(&pdev->dev, &hdmi_ops); +} + +static int imx_hdmi_platform_remove(struct platform_device *pdev) +{ +	component_del(&pdev->dev, &hdmi_ops); +	return 0; +} + +static struct platform_driver imx_hdmi_driver = { +	.probe  = imx_hdmi_platform_probe, +	.remove = imx_hdmi_platform_remove, +	.driver = { +		.name = "imx-hdmi", +		.owner = THIS_MODULE, +		.of_match_table = imx_hdmi_dt_ids, +	}, +}; + +module_platform_driver(imx_hdmi_driver); + +MODULE_AUTHOR("Sascha Hauer <s.hauer@pengutronix.de>"); +MODULE_DESCRIPTION("i.MX6 HDMI transmitter driver"); +MODULE_LICENSE("GPL"); +MODULE_ALIAS("platform:imx-hdmi"); diff --git a/drivers/staging/imx-drm/imx-hdmi.h b/drivers/staging/imx-drm/imx-hdmi.h new file mode 100644 index 00000000000..39b677689db --- /dev/null +++ b/drivers/staging/imx-drm/imx-hdmi.h @@ -0,0 +1,1032 @@ +/* + * Copyright (C) 2011 Freescale Semiconductor, Inc. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + */ + +#ifndef __IMX_HDMI_H__ +#define __IMX_HDMI_H__ + +/* Identification Registers */ +#define HDMI_DESIGN_ID                          0x0000 +#define HDMI_REVISION_ID                        0x0001 +#define HDMI_PRODUCT_ID0                        0x0002 +#define HDMI_PRODUCT_ID1                        0x0003 +#define HDMI_CONFIG0_ID                         0x0004 +#define HDMI_CONFIG1_ID                         0x0005 +#define HDMI_CONFIG2_ID                         0x0006 +#define HDMI_CONFIG3_ID                         0x0007 + +/* Interrupt Registers */ +#define HDMI_IH_FC_STAT0                        0x0100 +#define HDMI_IH_FC_STAT1                        0x0101 +#define HDMI_IH_FC_STAT2                        0x0102 +#define HDMI_IH_AS_STAT0                        0x0103 +#define HDMI_IH_PHY_STAT0                       0x0104 +#define HDMI_IH_I2CM_STAT0                      0x0105 +#define HDMI_IH_CEC_STAT0                       0x0106 +#define HDMI_IH_VP_STAT0                        0x0107 +#define HDMI_IH_I2CMPHY_STAT0                   0x0108 +#define HDMI_IH_AHBDMAAUD_STAT0                 0x0109 + +#define HDMI_IH_MUTE_FC_STAT0                   0x0180 +#define HDMI_IH_MUTE_FC_STAT1                   0x0181 +#define HDMI_IH_MUTE_FC_STAT2                   0x0182 +#define HDMI_IH_MUTE_AS_STAT0                   0x0183 +#define HDMI_IH_MUTE_PHY_STAT0                  0x0184 +#define HDMI_IH_MUTE_I2CM_STAT0                 0x0185 +#define HDMI_IH_MUTE_CEC_STAT0                  0x0186 +#define HDMI_IH_MUTE_VP_STAT0                   0x0187 +#define HDMI_IH_MUTE_I2CMPHY_STAT0              0x0188 +#define HDMI_IH_MUTE_AHBDMAAUD_STAT0            0x0189 +#define HDMI_IH_MUTE                            0x01FF + +/* Video Sample Registers */ +#define HDMI_TX_INVID0                          0x0200 +#define HDMI_TX_INSTUFFING                      0x0201 +#define HDMI_TX_GYDATA0                         0x0202 +#define HDMI_TX_GYDATA1                         0x0203 +#define HDMI_TX_RCRDATA0                        0x0204 +#define HDMI_TX_RCRDATA1                        0x0205 +#define HDMI_TX_BCBDATA0                        0x0206 +#define HDMI_TX_BCBDATA1                        0x0207 + +/* Video Packetizer Registers */ +#define HDMI_VP_STATUS                          0x0800 +#define HDMI_VP_PR_CD                           0x0801 +#define HDMI_VP_STUFF                           0x0802 +#define HDMI_VP_REMAP                           0x0803 +#define HDMI_VP_CONF                            0x0804 +#define HDMI_VP_STAT                            0x0805 +#define HDMI_VP_INT                             0x0806 +#define HDMI_VP_MASK                            0x0807 +#define HDMI_VP_POL                             0x0808 + +/* Frame Composer Registers */ +#define HDMI_FC_INVIDCONF                       0x1000 +#define HDMI_FC_INHACTV0                        0x1001 +#define HDMI_FC_INHACTV1                        0x1002 +#define HDMI_FC_INHBLANK0                       0x1003 +#define HDMI_FC_INHBLANK1                       0x1004 +#define HDMI_FC_INVACTV0                        0x1005 +#define HDMI_FC_INVACTV1                        0x1006 +#define HDMI_FC_INVBLANK                        0x1007 +#define HDMI_FC_HSYNCINDELAY0                   0x1008 +#define HDMI_FC_HSYNCINDELAY1                   0x1009 +#define HDMI_FC_HSYNCINWIDTH0                   0x100A +#define HDMI_FC_HSYNCINWIDTH1                   0x100B +#define HDMI_FC_VSYNCINDELAY                    0x100C +#define HDMI_FC_VSYNCINWIDTH                    0x100D +#define HDMI_FC_INFREQ0                         0x100E +#define HDMI_FC_INFREQ1                         0x100F +#define HDMI_FC_INFREQ2                         0x1010 +#define HDMI_FC_CTRLDUR                         0x1011 +#define HDMI_FC_EXCTRLDUR                       0x1012 +#define HDMI_FC_EXCTRLSPAC                      0x1013 +#define HDMI_FC_CH0PREAM                        0x1014 +#define HDMI_FC_CH1PREAM                        0x1015 +#define HDMI_FC_CH2PREAM                        0x1016 +#define HDMI_FC_AVICONF3                        0x1017 +#define HDMI_FC_GCP                             0x1018 +#define HDMI_FC_AVICONF0                        0x1019 +#define HDMI_FC_AVICONF1                        0x101A +#define HDMI_FC_AVICONF2                        0x101B +#define HDMI_FC_AVIVID                          0x101C +#define HDMI_FC_AVIETB0                         0x101D +#define HDMI_FC_AVIETB1                         0x101E +#define HDMI_FC_AVISBB0                         0x101F +#define HDMI_FC_AVISBB1                         0x1020 +#define HDMI_FC_AVIELB0                         0x1021 +#define HDMI_FC_AVIELB1                         0x1022 +#define HDMI_FC_AVISRB0                         0x1023 +#define HDMI_FC_AVISRB1                         0x1024 +#define HDMI_FC_AUDICONF0                       0x1025 +#define HDMI_FC_AUDICONF1                       0x1026 +#define HDMI_FC_AUDICONF2                       0x1027 +#define HDMI_FC_AUDICONF3                       0x1028 +#define HDMI_FC_VSDIEEEID0                      0x1029 +#define HDMI_FC_VSDSIZE                         0x102A +#define HDMI_FC_VSDIEEEID1                      0x1030 +#define HDMI_FC_VSDIEEEID2                      0x1031 +#define HDMI_FC_VSDPAYLOAD0                     0x1032 +#define HDMI_FC_VSDPAYLOAD1                     0x1033 +#define HDMI_FC_VSDPAYLOAD2                     0x1034 +#define HDMI_FC_VSDPAYLOAD3                     0x1035 +#define HDMI_FC_VSDPAYLOAD4                     0x1036 +#define HDMI_FC_VSDPAYLOAD5                     0x1037 +#define HDMI_FC_VSDPAYLOAD6                     0x1038 +#define HDMI_FC_VSDPAYLOAD7                     0x1039 +#define HDMI_FC_VSDPAYLOAD8                     0x103A +#define HDMI_FC_VSDPAYLOAD9                     0x103B +#define HDMI_FC_VSDPAYLOAD10                    0x103C +#define HDMI_FC_VSDPAYLOAD11                    0x103D +#define HDMI_FC_VSDPAYLOAD12                    0x103E +#define HDMI_FC_VSDPAYLOAD13                    0x103F +#define HDMI_FC_VSDPAYLOAD14                    0x1040 +#define HDMI_FC_VSDPAYLOAD15                    0x1041 +#define HDMI_FC_VSDPAYLOAD16                    0x1042 +#define HDMI_FC_VSDPAYLOAD17                    0x1043 +#define HDMI_FC_VSDPAYLOAD18                    0x1044 +#define HDMI_FC_VSDPAYLOAD19                    0x1045 +#define HDMI_FC_VSDPAYLOAD20                    0x1046 +#define HDMI_FC_VSDPAYLOAD21                    0x1047 +#define HDMI_FC_VSDPAYLOAD22                    0x1048 +#define HDMI_FC_VSDPAYLOAD23                    0x1049 +#define HDMI_FC_SPDVENDORNAME0                  0x104A +#define HDMI_FC_SPDVENDORNAME1                  0x104B +#define HDMI_FC_SPDVENDORNAME2                  0x104C +#define HDMI_FC_SPDVENDORNAME3                  0x104D +#define HDMI_FC_SPDVENDORNAME4                  0x104E +#define HDMI_FC_SPDVENDORNAME5                  0x104F +#define HDMI_FC_SPDVENDORNAME6                  0x1050 +#define HDMI_FC_SPDVENDORNAME7                  0x1051 +#define HDMI_FC_SDPPRODUCTNAME0                 0x1052 +#define HDMI_FC_SDPPRODUCTNAME1                 0x1053 +#define HDMI_FC_SDPPRODUCTNAME2                 0x1054 +#define HDMI_FC_SDPPRODUCTNAME3                 0x1055 +#define HDMI_FC_SDPPRODUCTNAME4                 0x1056 +#define HDMI_FC_SDPPRODUCTNAME5                 0x1057 +#define HDMI_FC_SDPPRODUCTNAME6                 0x1058 +#define HDMI_FC_SDPPRODUCTNAME7                 0x1059 +#define HDMI_FC_SDPPRODUCTNAME8                 0x105A +#define HDMI_FC_SDPPRODUCTNAME9                 0x105B +#define HDMI_FC_SDPPRODUCTNAME10                0x105C +#define HDMI_FC_SDPPRODUCTNAME11                0x105D +#define HDMI_FC_SDPPRODUCTNAME12                0x105E +#define HDMI_FC_SDPPRODUCTNAME13                0x105F +#define HDMI_FC_SDPPRODUCTNAME14                0x1060 +#define HDMI_FC_SPDPRODUCTNAME15                0x1061 +#define HDMI_FC_SPDDEVICEINF                    0x1062 +#define HDMI_FC_AUDSCONF                        0x1063 +#define HDMI_FC_AUDSSTAT                        0x1064 +#define HDMI_FC_DATACH0FILL                     0x1070 +#define HDMI_FC_DATACH1FILL                     0x1071 +#define HDMI_FC_DATACH2FILL                     0x1072 +#define HDMI_FC_CTRLQHIGH                       0x1073 +#define HDMI_FC_CTRLQLOW                        0x1074 +#define HDMI_FC_ACP0                            0x1075 +#define HDMI_FC_ACP28                           0x1076 +#define HDMI_FC_ACP27                           0x1077 +#define HDMI_FC_ACP26                           0x1078 +#define HDMI_FC_ACP25                           0x1079 +#define HDMI_FC_ACP24                           0x107A +#define HDMI_FC_ACP23                           0x107B +#define HDMI_FC_ACP22                           0x107C +#define HDMI_FC_ACP21                           0x107D +#define HDMI_FC_ACP20                           0x107E +#define HDMI_FC_ACP19                           0x107F +#define HDMI_FC_ACP18                           0x1080 +#define HDMI_FC_ACP17                           0x1081 +#define HDMI_FC_ACP16                           0x1082 +#define HDMI_FC_ACP15                           0x1083 +#define HDMI_FC_ACP14                           0x1084 +#define HDMI_FC_ACP13                           0x1085 +#define HDMI_FC_ACP12                           0x1086 +#define HDMI_FC_ACP11                           0x1087 +#define HDMI_FC_ACP10                           0x1088 +#define HDMI_FC_ACP9                            0x1089 +#define HDMI_FC_ACP8                            0x108A +#define HDMI_FC_ACP7                            0x108B +#define HDMI_FC_ACP6                            0x108C +#define HDMI_FC_ACP5                            0x108D +#define HDMI_FC_ACP4                            0x108E +#define HDMI_FC_ACP3                            0x108F +#define HDMI_FC_ACP2                            0x1090 +#define HDMI_FC_ACP1                            0x1091 +#define HDMI_FC_ISCR1_0                         0x1092 +#define HDMI_FC_ISCR1_16                        0x1093 +#define HDMI_FC_ISCR1_15                        0x1094 +#define HDMI_FC_ISCR1_14                        0x1095 +#define HDMI_FC_ISCR1_13                        0x1096 +#define HDMI_FC_ISCR1_12                        0x1097 +#define HDMI_FC_ISCR1_11                        0x1098 +#define HDMI_FC_ISCR1_10                        0x1099 +#define HDMI_FC_ISCR1_9                         0x109A +#define HDMI_FC_ISCR1_8                         0x109B +#define HDMI_FC_ISCR1_7                         0x109C +#define HDMI_FC_ISCR1_6                         0x109D +#define HDMI_FC_ISCR1_5                         0x109E +#define HDMI_FC_ISCR1_4                         0x109F +#define HDMI_FC_ISCR1_3                         0x10A0 +#define HDMI_FC_ISCR1_2                         0x10A1 +#define HDMI_FC_ISCR1_1                         0x10A2 +#define HDMI_FC_ISCR2_15                        0x10A3 +#define HDMI_FC_ISCR2_14                        0x10A4 +#define HDMI_FC_ISCR2_13                        0x10A5 +#define HDMI_FC_ISCR2_12                        0x10A6 +#define HDMI_FC_ISCR2_11                        0x10A7 +#define HDMI_FC_ISCR2_10                        0x10A8 +#define HDMI_FC_ISCR2_9                         0x10A9 +#define HDMI_FC_ISCR2_8                         0x10AA +#define HDMI_FC_ISCR2_7                         0x10AB +#define HDMI_FC_ISCR2_6                         0x10AC +#define HDMI_FC_ISCR2_5                         0x10AD +#define HDMI_FC_ISCR2_4                         0x10AE +#define HDMI_FC_ISCR2_3                         0x10AF +#define HDMI_FC_ISCR2_2                         0x10B0 +#define HDMI_FC_ISCR2_1                         0x10B1 +#define HDMI_FC_ISCR2_0                         0x10B2 +#define HDMI_FC_DATAUTO0                        0x10B3 +#define HDMI_FC_DATAUTO1                        0x10B4 +#define HDMI_FC_DATAUTO2                        0x10B5 +#define HDMI_FC_DATMAN                          0x10B6 +#define HDMI_FC_DATAUTO3                        0x10B7 +#define HDMI_FC_RDRB0                           0x10B8 +#define HDMI_FC_RDRB1                           0x10B9 +#define HDMI_FC_RDRB2                           0x10BA +#define HDMI_FC_RDRB3                           0x10BB +#define HDMI_FC_RDRB4                           0x10BC +#define HDMI_FC_RDRB5                           0x10BD +#define HDMI_FC_RDRB6                           0x10BE +#define HDMI_FC_RDRB7                           0x10BF +#define HDMI_FC_STAT0                           0x10D0 +#define HDMI_FC_INT0                            0x10D1 +#define HDMI_FC_MASK0                           0x10D2 +#define HDMI_FC_POL0                            0x10D3 +#define HDMI_FC_STAT1                           0x10D4 +#define HDMI_FC_INT1                            0x10D5 +#define HDMI_FC_MASK1                           0x10D6 +#define HDMI_FC_POL1                            0x10D7 +#define HDMI_FC_STAT2                           0x10D8 +#define HDMI_FC_INT2                            0x10D9 +#define HDMI_FC_MASK2                           0x10DA +#define HDMI_FC_POL2                            0x10DB +#define HDMI_FC_PRCONF                          0x10E0 + +#define HDMI_FC_GMD_STAT                        0x1100 +#define HDMI_FC_GMD_EN                          0x1101 +#define HDMI_FC_GMD_UP                          0x1102 +#define HDMI_FC_GMD_CONF                        0x1103 +#define HDMI_FC_GMD_HB                          0x1104 +#define HDMI_FC_GMD_PB0                         0x1105 +#define HDMI_FC_GMD_PB1                         0x1106 +#define HDMI_FC_GMD_PB2                         0x1107 +#define HDMI_FC_GMD_PB3                         0x1108 +#define HDMI_FC_GMD_PB4                         0x1109 +#define HDMI_FC_GMD_PB5                         0x110A +#define HDMI_FC_GMD_PB6                         0x110B +#define HDMI_FC_GMD_PB7                         0x110C +#define HDMI_FC_GMD_PB8                         0x110D +#define HDMI_FC_GMD_PB9                         0x110E +#define HDMI_FC_GMD_PB10                        0x110F +#define HDMI_FC_GMD_PB11                        0x1110 +#define HDMI_FC_GMD_PB12                        0x1111 +#define HDMI_FC_GMD_PB13                        0x1112 +#define HDMI_FC_GMD_PB14                        0x1113 +#define HDMI_FC_GMD_PB15                        0x1114 +#define HDMI_FC_GMD_PB16                        0x1115 +#define HDMI_FC_GMD_PB17                        0x1116 +#define HDMI_FC_GMD_PB18                        0x1117 +#define HDMI_FC_GMD_PB19                        0x1118 +#define HDMI_FC_GMD_PB20                        0x1119 +#define HDMI_FC_GMD_PB21                        0x111A +#define HDMI_FC_GMD_PB22                        0x111B +#define HDMI_FC_GMD_PB23                        0x111C +#define HDMI_FC_GMD_PB24                        0x111D +#define HDMI_FC_GMD_PB25                        0x111E +#define HDMI_FC_GMD_PB26                        0x111F +#define HDMI_FC_GMD_PB27                        0x1120 + +#define HDMI_FC_DBGFORCE                        0x1200 +#define HDMI_FC_DBGAUD0CH0                      0x1201 +#define HDMI_FC_DBGAUD1CH0                      0x1202 +#define HDMI_FC_DBGAUD2CH0                      0x1203 +#define HDMI_FC_DBGAUD0CH1                      0x1204 +#define HDMI_FC_DBGAUD1CH1                      0x1205 +#define HDMI_FC_DBGAUD2CH1                      0x1206 +#define HDMI_FC_DBGAUD0CH2                      0x1207 +#define HDMI_FC_DBGAUD1CH2                      0x1208 +#define HDMI_FC_DBGAUD2CH2                      0x1209 +#define HDMI_FC_DBGAUD0CH3                      0x120A +#define HDMI_FC_DBGAUD1CH3                      0x120B +#define HDMI_FC_DBGAUD2CH3                      0x120C +#define HDMI_FC_DBGAUD0CH4                      0x120D +#define HDMI_FC_DBGAUD1CH4                      0x120E +#define HDMI_FC_DBGAUD2CH4                      0x120F +#define HDMI_FC_DBGAUD0CH5                      0x1210 +#define HDMI_FC_DBGAUD1CH5                      0x1211 +#define HDMI_FC_DBGAUD2CH5                      0x1212 +#define HDMI_FC_DBGAUD0CH6                      0x1213 +#define HDMI_FC_DBGAUD1CH6                      0x1214 +#define HDMI_FC_DBGAUD2CH6                      0x1215 +#define HDMI_FC_DBGAUD0CH7                      0x1216 +#define HDMI_FC_DBGAUD1CH7                      0x1217 +#define HDMI_FC_DBGAUD2CH7                      0x1218 +#define HDMI_FC_DBGTMDS0                        0x1219 +#define HDMI_FC_DBGTMDS1                        0x121A +#define HDMI_FC_DBGTMDS2                        0x121B + +/* HDMI Source PHY Registers */ +#define HDMI_PHY_CONF0                          0x3000 +#define HDMI_PHY_TST0                           0x3001 +#define HDMI_PHY_TST1                           0x3002 +#define HDMI_PHY_TST2                           0x3003 +#define HDMI_PHY_STAT0                          0x3004 +#define HDMI_PHY_INT0                           0x3005 +#define HDMI_PHY_MASK0                          0x3006 +#define HDMI_PHY_POL0                           0x3007 + +/* HDMI Master PHY Registers */ +#define HDMI_PHY_I2CM_SLAVE_ADDR                0x3020 +#define HDMI_PHY_I2CM_ADDRESS_ADDR              0x3021 +#define HDMI_PHY_I2CM_DATAO_1_ADDR              0x3022 +#define HDMI_PHY_I2CM_DATAO_0_ADDR              0x3023 +#define HDMI_PHY_I2CM_DATAI_1_ADDR              0x3024 +#define HDMI_PHY_I2CM_DATAI_0_ADDR              0x3025 +#define HDMI_PHY_I2CM_OPERATION_ADDR            0x3026 +#define HDMI_PHY_I2CM_INT_ADDR                  0x3027 +#define HDMI_PHY_I2CM_CTLINT_ADDR               0x3028 +#define HDMI_PHY_I2CM_DIV_ADDR                  0x3029 +#define HDMI_PHY_I2CM_SOFTRSTZ_ADDR             0x302a +#define HDMI_PHY_I2CM_SS_SCL_HCNT_1_ADDR        0x302b +#define HDMI_PHY_I2CM_SS_SCL_HCNT_0_ADDR        0x302c +#define HDMI_PHY_I2CM_SS_SCL_LCNT_1_ADDR        0x302d +#define HDMI_PHY_I2CM_SS_SCL_LCNT_0_ADDR        0x302e +#define HDMI_PHY_I2CM_FS_SCL_HCNT_1_ADDR        0x302f +#define HDMI_PHY_I2CM_FS_SCL_HCNT_0_ADDR        0x3030 +#define HDMI_PHY_I2CM_FS_SCL_LCNT_1_ADDR        0x3031 +#define HDMI_PHY_I2CM_FS_SCL_LCNT_0_ADDR        0x3032 + +/* Audio Sampler Registers */ +#define HDMI_AUD_CONF0                          0x3100 +#define HDMI_AUD_CONF1                          0x3101 +#define HDMI_AUD_INT                            0x3102 +#define HDMI_AUD_CONF2                          0x3103 +#define HDMI_AUD_N1                             0x3200 +#define HDMI_AUD_N2                             0x3201 +#define HDMI_AUD_N3                             0x3202 +#define HDMI_AUD_CTS1                           0x3203 +#define HDMI_AUD_CTS2                           0x3204 +#define HDMI_AUD_CTS3                           0x3205 +#define HDMI_AUD_INPUTCLKFS                     0x3206 +#define HDMI_AUD_SPDIFINT			0x3302 +#define HDMI_AUD_CONF0_HBR                      0x3400 +#define HDMI_AUD_HBR_STATUS                     0x3401 +#define HDMI_AUD_HBR_INT                        0x3402 +#define HDMI_AUD_HBR_POL                        0x3403 +#define HDMI_AUD_HBR_MASK                       0x3404 + +/* + * Generic Parallel Audio Interface Registers + * Not used as GPAUD interface is not enabled in hw + */ +#define HDMI_GP_CONF0                           0x3500 +#define HDMI_GP_CONF1                           0x3501 +#define HDMI_GP_CONF2                           0x3502 +#define HDMI_GP_STAT                            0x3503 +#define HDMI_GP_INT                             0x3504 +#define HDMI_GP_MASK                            0x3505 +#define HDMI_GP_POL                             0x3506 + +/* Audio DMA Registers */ +#define HDMI_AHB_DMA_CONF0                      0x3600 +#define HDMI_AHB_DMA_START                      0x3601 +#define HDMI_AHB_DMA_STOP                       0x3602 +#define HDMI_AHB_DMA_THRSLD                     0x3603 +#define HDMI_AHB_DMA_STRADDR0                   0x3604 +#define HDMI_AHB_DMA_STRADDR1                   0x3605 +#define HDMI_AHB_DMA_STRADDR2                   0x3606 +#define HDMI_AHB_DMA_STRADDR3                   0x3607 +#define HDMI_AHB_DMA_STPADDR0                   0x3608 +#define HDMI_AHB_DMA_STPADDR1                   0x3609 +#define HDMI_AHB_DMA_STPADDR2                   0x360a +#define HDMI_AHB_DMA_STPADDR3                   0x360b +#define HDMI_AHB_DMA_BSTADDR0                   0x360c +#define HDMI_AHB_DMA_BSTADDR1                   0x360d +#define HDMI_AHB_DMA_BSTADDR2                   0x360e +#define HDMI_AHB_DMA_BSTADDR3                   0x360f +#define HDMI_AHB_DMA_MBLENGTH0                  0x3610 +#define HDMI_AHB_DMA_MBLENGTH1                  0x3611 +#define HDMI_AHB_DMA_STAT                       0x3612 +#define HDMI_AHB_DMA_INT                        0x3613 +#define HDMI_AHB_DMA_MASK                       0x3614 +#define HDMI_AHB_DMA_POL                        0x3615 +#define HDMI_AHB_DMA_CONF1                      0x3616 +#define HDMI_AHB_DMA_BUFFSTAT                   0x3617 +#define HDMI_AHB_DMA_BUFFINT                    0x3618 +#define HDMI_AHB_DMA_BUFFMASK                   0x3619 +#define HDMI_AHB_DMA_BUFFPOL                    0x361a + +/* Main Controller Registers */ +#define HDMI_MC_SFRDIV                          0x4000 +#define HDMI_MC_CLKDIS                          0x4001 +#define HDMI_MC_SWRSTZ                          0x4002 +#define HDMI_MC_OPCTRL                          0x4003 +#define HDMI_MC_FLOWCTRL                        0x4004 +#define HDMI_MC_PHYRSTZ                         0x4005 +#define HDMI_MC_LOCKONCLOCK                     0x4006 +#define HDMI_MC_HEACPHY_RST                     0x4007 + +/* Color Space  Converter Registers */ +#define HDMI_CSC_CFG                            0x4100 +#define HDMI_CSC_SCALE                          0x4101 +#define HDMI_CSC_COEF_A1_MSB                    0x4102 +#define HDMI_CSC_COEF_A1_LSB                    0x4103 +#define HDMI_CSC_COEF_A2_MSB                    0x4104 +#define HDMI_CSC_COEF_A2_LSB                    0x4105 +#define HDMI_CSC_COEF_A3_MSB                    0x4106 +#define HDMI_CSC_COEF_A3_LSB                    0x4107 +#define HDMI_CSC_COEF_A4_MSB                    0x4108 +#define HDMI_CSC_COEF_A4_LSB                    0x4109 +#define HDMI_CSC_COEF_B1_MSB                    0x410A +#define HDMI_CSC_COEF_B1_LSB                    0x410B +#define HDMI_CSC_COEF_B2_MSB                    0x410C +#define HDMI_CSC_COEF_B2_LSB                    0x410D +#define HDMI_CSC_COEF_B3_MSB                    0x410E +#define HDMI_CSC_COEF_B3_LSB                    0x410F +#define HDMI_CSC_COEF_B4_MSB                    0x4110 +#define HDMI_CSC_COEF_B4_LSB                    0x4111 +#define HDMI_CSC_COEF_C1_MSB                    0x4112 +#define HDMI_CSC_COEF_C1_LSB                    0x4113 +#define HDMI_CSC_COEF_C2_MSB                    0x4114 +#define HDMI_CSC_COEF_C2_LSB                    0x4115 +#define HDMI_CSC_COEF_C3_MSB                    0x4116 +#define HDMI_CSC_COEF_C3_LSB                    0x4117 +#define HDMI_CSC_COEF_C4_MSB                    0x4118 +#define HDMI_CSC_COEF_C4_LSB                    0x4119 + +/* HDCP Encryption Engine Registers */ +#define HDMI_A_HDCPCFG0                         0x5000 +#define HDMI_A_HDCPCFG1                         0x5001 +#define HDMI_A_HDCPOBS0                         0x5002 +#define HDMI_A_HDCPOBS1                         0x5003 +#define HDMI_A_HDCPOBS2                         0x5004 +#define HDMI_A_HDCPOBS3                         0x5005 +#define HDMI_A_APIINTCLR                        0x5006 +#define HDMI_A_APIINTSTAT                       0x5007 +#define HDMI_A_APIINTMSK                        0x5008 +#define HDMI_A_VIDPOLCFG                        0x5009 +#define HDMI_A_OESSWCFG                         0x500A +#define HDMI_A_TIMER1SETUP0                     0x500B +#define HDMI_A_TIMER1SETUP1                     0x500C +#define HDMI_A_TIMER2SETUP0                     0x500D +#define HDMI_A_TIMER2SETUP1                     0x500E +#define HDMI_A_100MSCFG                         0x500F +#define HDMI_A_2SCFG0                           0x5010 +#define HDMI_A_2SCFG1                           0x5011 +#define HDMI_A_5SCFG0                           0x5012 +#define HDMI_A_5SCFG1                           0x5013 +#define HDMI_A_SRMVERLSB                        0x5014 +#define HDMI_A_SRMVERMSB                        0x5015 +#define HDMI_A_SRMCTRL                          0x5016 +#define HDMI_A_SFRSETUP                         0x5017 +#define HDMI_A_I2CHSETUP                        0x5018 +#define HDMI_A_INTSETUP                         0x5019 +#define HDMI_A_PRESETUP                         0x501A +#define HDMI_A_SRM_BASE                         0x5020 + +/* CEC Engine Registers */ +#define HDMI_CEC_CTRL                           0x7D00 +#define HDMI_CEC_STAT                           0x7D01 +#define HDMI_CEC_MASK                           0x7D02 +#define HDMI_CEC_POLARITY                       0x7D03 +#define HDMI_CEC_INT                            0x7D04 +#define HDMI_CEC_ADDR_L                         0x7D05 +#define HDMI_CEC_ADDR_H                         0x7D06 +#define HDMI_CEC_TX_CNT                         0x7D07 +#define HDMI_CEC_RX_CNT                         0x7D08 +#define HDMI_CEC_TX_DATA0                       0x7D10 +#define HDMI_CEC_TX_DATA1                       0x7D11 +#define HDMI_CEC_TX_DATA2                       0x7D12 +#define HDMI_CEC_TX_DATA3                       0x7D13 +#define HDMI_CEC_TX_DATA4                       0x7D14 +#define HDMI_CEC_TX_DATA5                       0x7D15 +#define HDMI_CEC_TX_DATA6                       0x7D16 +#define HDMI_CEC_TX_DATA7                       0x7D17 +#define HDMI_CEC_TX_DATA8                       0x7D18 +#define HDMI_CEC_TX_DATA9                       0x7D19 +#define HDMI_CEC_TX_DATA10                      0x7D1a +#define HDMI_CEC_TX_DATA11                      0x7D1b +#define HDMI_CEC_TX_DATA12                      0x7D1c +#define HDMI_CEC_TX_DATA13                      0x7D1d +#define HDMI_CEC_TX_DATA14                      0x7D1e +#define HDMI_CEC_TX_DATA15                      0x7D1f +#define HDMI_CEC_RX_DATA0                       0x7D20 +#define HDMI_CEC_RX_DATA1                       0x7D21 +#define HDMI_CEC_RX_DATA2                       0x7D22 +#define HDMI_CEC_RX_DATA3                       0x7D23 +#define HDMI_CEC_RX_DATA4                       0x7D24 +#define HDMI_CEC_RX_DATA5                       0x7D25 +#define HDMI_CEC_RX_DATA6                       0x7D26 +#define HDMI_CEC_RX_DATA7                       0x7D27 +#define HDMI_CEC_RX_DATA8                       0x7D28 +#define HDMI_CEC_RX_DATA9                       0x7D29 +#define HDMI_CEC_RX_DATA10                      0x7D2a +#define HDMI_CEC_RX_DATA11                      0x7D2b +#define HDMI_CEC_RX_DATA12                      0x7D2c +#define HDMI_CEC_RX_DATA13                      0x7D2d +#define HDMI_CEC_RX_DATA14                      0x7D2e +#define HDMI_CEC_RX_DATA15                      0x7D2f +#define HDMI_CEC_LOCK                           0x7D30 +#define HDMI_CEC_WKUPCTRL                       0x7D31 + +/* I2C Master Registers (E-DDC) */ +#define HDMI_I2CM_SLAVE                         0x7E00 +#define HDMI_I2CMESS                            0x7E01 +#define HDMI_I2CM_DATAO                         0x7E02 +#define HDMI_I2CM_DATAI                         0x7E03 +#define HDMI_I2CM_OPERATION                     0x7E04 +#define HDMI_I2CM_INT                           0x7E05 +#define HDMI_I2CM_CTLINT                        0x7E06 +#define HDMI_I2CM_DIV                           0x7E07 +#define HDMI_I2CM_SEGADDR                       0x7E08 +#define HDMI_I2CM_SOFTRSTZ                      0x7E09 +#define HDMI_I2CM_SEGPTR                        0x7E0A +#define HDMI_I2CM_SS_SCL_HCNT_1_ADDR            0x7E0B +#define HDMI_I2CM_SS_SCL_HCNT_0_ADDR            0x7E0C +#define HDMI_I2CM_SS_SCL_LCNT_1_ADDR            0x7E0D +#define HDMI_I2CM_SS_SCL_LCNT_0_ADDR            0x7E0E +#define HDMI_I2CM_FS_SCL_HCNT_1_ADDR            0x7E0F +#define HDMI_I2CM_FS_SCL_HCNT_0_ADDR            0x7E10 +#define HDMI_I2CM_FS_SCL_LCNT_1_ADDR            0x7E11 +#define HDMI_I2CM_FS_SCL_LCNT_0_ADDR            0x7E12 + +enum { +/* IH_FC_INT2 field values */ +	HDMI_IH_FC_INT2_OVERFLOW_MASK = 0x03, +	HDMI_IH_FC_INT2_LOW_PRIORITY_OVERFLOW = 0x02, +	HDMI_IH_FC_INT2_HIGH_PRIORITY_OVERFLOW = 0x01, + +/* IH_FC_STAT2 field values */ +	HDMI_IH_FC_STAT2_OVERFLOW_MASK = 0x03, +	HDMI_IH_FC_STAT2_LOW_PRIORITY_OVERFLOW = 0x02, +	HDMI_IH_FC_STAT2_HIGH_PRIORITY_OVERFLOW = 0x01, + +/* IH_PHY_STAT0 field values */ +	HDMI_IH_PHY_STAT0_RX_SENSE3 = 0x20, +	HDMI_IH_PHY_STAT0_RX_SENSE2 = 0x10, +	HDMI_IH_PHY_STAT0_RX_SENSE1 = 0x8, +	HDMI_IH_PHY_STAT0_RX_SENSE0 = 0x4, +	HDMI_IH_PHY_STAT0_TX_PHY_LOCK = 0x2, +	HDMI_IH_PHY_STAT0_HPD = 0x1, + +/* IH_MUTE_I2CMPHY_STAT0 field values */ +	HDMI_IH_MUTE_I2CMPHY_STAT0_I2CMPHYDONE = 0x2, +	HDMI_IH_MUTE_I2CMPHY_STAT0_I2CMPHYERROR = 0x1, + +/* IH_AHBDMAAUD_STAT0 field values */ +	HDMI_IH_AHBDMAAUD_STAT0_ERROR = 0x20, +	HDMI_IH_AHBDMAAUD_STAT0_LOST = 0x10, +	HDMI_IH_AHBDMAAUD_STAT0_RETRY = 0x08, +	HDMI_IH_AHBDMAAUD_STAT0_DONE = 0x04, +	HDMI_IH_AHBDMAAUD_STAT0_BUFFFULL = 0x02, +	HDMI_IH_AHBDMAAUD_STAT0_BUFFEMPTY = 0x01, + +/* IH_MUTE_FC_STAT2 field values */ +	HDMI_IH_MUTE_FC_STAT2_OVERFLOW_MASK = 0x03, +	HDMI_IH_MUTE_FC_STAT2_LOW_PRIORITY_OVERFLOW = 0x02, +	HDMI_IH_MUTE_FC_STAT2_HIGH_PRIORITY_OVERFLOW = 0x01, + +/* IH_MUTE_AHBDMAAUD_STAT0 field values */ +	HDMI_IH_MUTE_AHBDMAAUD_STAT0_ERROR = 0x20, +	HDMI_IH_MUTE_AHBDMAAUD_STAT0_LOST = 0x10, +	HDMI_IH_MUTE_AHBDMAAUD_STAT0_RETRY = 0x08, +	HDMI_IH_MUTE_AHBDMAAUD_STAT0_DONE = 0x04, +	HDMI_IH_MUTE_AHBDMAAUD_STAT0_BUFFFULL = 0x02, +	HDMI_IH_MUTE_AHBDMAAUD_STAT0_BUFFEMPTY = 0x01, + +/* IH_MUTE field values */ +	HDMI_IH_MUTE_MUTE_WAKEUP_INTERRUPT = 0x2, +	HDMI_IH_MUTE_MUTE_ALL_INTERRUPT = 0x1, + +/* TX_INVID0 field values */ +	HDMI_TX_INVID0_INTERNAL_DE_GENERATOR_MASK = 0x80, +	HDMI_TX_INVID0_INTERNAL_DE_GENERATOR_ENABLE = 0x80, +	HDMI_TX_INVID0_INTERNAL_DE_GENERATOR_DISABLE = 0x00, +	HDMI_TX_INVID0_VIDEO_MAPPING_MASK = 0x1F, +	HDMI_TX_INVID0_VIDEO_MAPPING_OFFSET = 0, + +/* TX_INSTUFFING field values */ +	HDMI_TX_INSTUFFING_BDBDATA_STUFFING_MASK = 0x4, +	HDMI_TX_INSTUFFING_BDBDATA_STUFFING_ENABLE = 0x4, +	HDMI_TX_INSTUFFING_BDBDATA_STUFFING_DISABLE = 0x0, +	HDMI_TX_INSTUFFING_RCRDATA_STUFFING_MASK = 0x2, +	HDMI_TX_INSTUFFING_RCRDATA_STUFFING_ENABLE = 0x2, +	HDMI_TX_INSTUFFING_RCRDATA_STUFFING_DISABLE = 0x0, +	HDMI_TX_INSTUFFING_GYDATA_STUFFING_MASK = 0x1, +	HDMI_TX_INSTUFFING_GYDATA_STUFFING_ENABLE = 0x1, +	HDMI_TX_INSTUFFING_GYDATA_STUFFING_DISABLE = 0x0, + +/* VP_PR_CD field values */ +	HDMI_VP_PR_CD_COLOR_DEPTH_MASK = 0xF0, +	HDMI_VP_PR_CD_COLOR_DEPTH_OFFSET = 4, +	HDMI_VP_PR_CD_DESIRED_PR_FACTOR_MASK = 0x0F, +	HDMI_VP_PR_CD_DESIRED_PR_FACTOR_OFFSET = 0, + +/* VP_STUFF field values */ +	HDMI_VP_STUFF_IDEFAULT_PHASE_MASK = 0x20, +	HDMI_VP_STUFF_IDEFAULT_PHASE_OFFSET = 5, +	HDMI_VP_STUFF_IFIX_PP_TO_LAST_MASK = 0x10, +	HDMI_VP_STUFF_IFIX_PP_TO_LAST_OFFSET = 4, +	HDMI_VP_STUFF_ICX_GOTO_P0_ST_MASK = 0x8, +	HDMI_VP_STUFF_ICX_GOTO_P0_ST_OFFSET = 3, +	HDMI_VP_STUFF_YCC422_STUFFING_MASK = 0x4, +	HDMI_VP_STUFF_YCC422_STUFFING_STUFFING_MODE = 0x4, +	HDMI_VP_STUFF_YCC422_STUFFING_DIRECT_MODE = 0x0, +	HDMI_VP_STUFF_PP_STUFFING_MASK = 0x2, +	HDMI_VP_STUFF_PP_STUFFING_STUFFING_MODE = 0x2, +	HDMI_VP_STUFF_PP_STUFFING_DIRECT_MODE = 0x0, +	HDMI_VP_STUFF_PR_STUFFING_MASK = 0x1, +	HDMI_VP_STUFF_PR_STUFFING_STUFFING_MODE = 0x1, +	HDMI_VP_STUFF_PR_STUFFING_DIRECT_MODE = 0x0, + +/* VP_CONF field values */ +	HDMI_VP_CONF_BYPASS_EN_MASK = 0x40, +	HDMI_VP_CONF_BYPASS_EN_ENABLE = 0x40, +	HDMI_VP_CONF_BYPASS_EN_DISABLE = 0x00, +	HDMI_VP_CONF_PP_EN_ENMASK = 0x20, +	HDMI_VP_CONF_PP_EN_ENABLE = 0x20, +	HDMI_VP_CONF_PP_EN_DISABLE = 0x00, +	HDMI_VP_CONF_PR_EN_MASK = 0x10, +	HDMI_VP_CONF_PR_EN_ENABLE = 0x10, +	HDMI_VP_CONF_PR_EN_DISABLE = 0x00, +	HDMI_VP_CONF_YCC422_EN_MASK = 0x8, +	HDMI_VP_CONF_YCC422_EN_ENABLE = 0x8, +	HDMI_VP_CONF_YCC422_EN_DISABLE = 0x0, +	HDMI_VP_CONF_BYPASS_SELECT_MASK = 0x4, +	HDMI_VP_CONF_BYPASS_SELECT_VID_PACKETIZER = 0x4, +	HDMI_VP_CONF_BYPASS_SELECT_PIX_REPEATER = 0x0, +	HDMI_VP_CONF_OUTPUT_SELECTOR_MASK = 0x3, +	HDMI_VP_CONF_OUTPUT_SELECTOR_BYPASS = 0x3, +	HDMI_VP_CONF_OUTPUT_SELECTOR_YCC422 = 0x1, +	HDMI_VP_CONF_OUTPUT_SELECTOR_PP = 0x0, + +/* VP_REMAP field values */ +	HDMI_VP_REMAP_MASK = 0x3, +	HDMI_VP_REMAP_YCC422_24bit = 0x2, +	HDMI_VP_REMAP_YCC422_20bit = 0x1, +	HDMI_VP_REMAP_YCC422_16bit = 0x0, + +/* FC_INVIDCONF field values */ +	HDMI_FC_INVIDCONF_HDCP_KEEPOUT_MASK = 0x80, +	HDMI_FC_INVIDCONF_HDCP_KEEPOUT_ACTIVE = 0x80, +	HDMI_FC_INVIDCONF_HDCP_KEEPOUT_INACTIVE = 0x00, +	HDMI_FC_INVIDCONF_VSYNC_IN_POLARITY_MASK = 0x40, +	HDMI_FC_INVIDCONF_VSYNC_IN_POLARITY_ACTIVE_HIGH = 0x40, +	HDMI_FC_INVIDCONF_VSYNC_IN_POLARITY_ACTIVE_LOW = 0x00, +	HDMI_FC_INVIDCONF_HSYNC_IN_POLARITY_MASK = 0x20, +	HDMI_FC_INVIDCONF_HSYNC_IN_POLARITY_ACTIVE_HIGH = 0x20, +	HDMI_FC_INVIDCONF_HSYNC_IN_POLARITY_ACTIVE_LOW = 0x00, +	HDMI_FC_INVIDCONF_DE_IN_POLARITY_MASK = 0x10, +	HDMI_FC_INVIDCONF_DE_IN_POLARITY_ACTIVE_HIGH = 0x10, +	HDMI_FC_INVIDCONF_DE_IN_POLARITY_ACTIVE_LOW = 0x00, +	HDMI_FC_INVIDCONF_DVI_MODEZ_MASK = 0x8, +	HDMI_FC_INVIDCONF_DVI_MODEZ_HDMI_MODE = 0x8, +	HDMI_FC_INVIDCONF_DVI_MODEZ_DVI_MODE = 0x0, +	HDMI_FC_INVIDCONF_R_V_BLANK_IN_OSC_MASK = 0x2, +	HDMI_FC_INVIDCONF_R_V_BLANK_IN_OSC_ACTIVE_HIGH = 0x2, +	HDMI_FC_INVIDCONF_R_V_BLANK_IN_OSC_ACTIVE_LOW = 0x0, +	HDMI_FC_INVIDCONF_IN_I_P_MASK = 0x1, +	HDMI_FC_INVIDCONF_IN_I_P_INTERLACED = 0x1, +	HDMI_FC_INVIDCONF_IN_I_P_PROGRESSIVE = 0x0, + +/* FC_AUDICONF0 field values */ +	HDMI_FC_AUDICONF0_CC_OFFSET = 4, +	HDMI_FC_AUDICONF0_CC_MASK = 0x70, +	HDMI_FC_AUDICONF0_CT_OFFSET = 0, +	HDMI_FC_AUDICONF0_CT_MASK = 0xF, + +/* FC_AUDICONF1 field values */ +	HDMI_FC_AUDICONF1_SS_OFFSET = 3, +	HDMI_FC_AUDICONF1_SS_MASK = 0x18, +	HDMI_FC_AUDICONF1_SF_OFFSET = 0, +	HDMI_FC_AUDICONF1_SF_MASK = 0x7, + +/* FC_AUDICONF3 field values */ +	HDMI_FC_AUDICONF3_LFEPBL_OFFSET = 5, +	HDMI_FC_AUDICONF3_LFEPBL_MASK = 0x60, +	HDMI_FC_AUDICONF3_DM_INH_OFFSET = 4, +	HDMI_FC_AUDICONF3_DM_INH_MASK = 0x10, +	HDMI_FC_AUDICONF3_LSV_OFFSET = 0, +	HDMI_FC_AUDICONF3_LSV_MASK = 0xF, + +/* FC_AUDSCHNLS0 field values */ +	HDMI_FC_AUDSCHNLS0_CGMSA_OFFSET = 4, +	HDMI_FC_AUDSCHNLS0_CGMSA_MASK = 0x30, +	HDMI_FC_AUDSCHNLS0_COPYRIGHT_OFFSET = 0, +	HDMI_FC_AUDSCHNLS0_COPYRIGHT_MASK = 0x01, + +/* FC_AUDSCHNLS3-6 field values */ +	HDMI_FC_AUDSCHNLS3_OIEC_CH0_OFFSET = 0, +	HDMI_FC_AUDSCHNLS3_OIEC_CH0_MASK = 0x0f, +	HDMI_FC_AUDSCHNLS3_OIEC_CH1_OFFSET = 4, +	HDMI_FC_AUDSCHNLS3_OIEC_CH1_MASK = 0xf0, +	HDMI_FC_AUDSCHNLS4_OIEC_CH2_OFFSET = 0, +	HDMI_FC_AUDSCHNLS4_OIEC_CH2_MASK = 0x0f, +	HDMI_FC_AUDSCHNLS4_OIEC_CH3_OFFSET = 4, +	HDMI_FC_AUDSCHNLS4_OIEC_CH3_MASK = 0xf0, + +	HDMI_FC_AUDSCHNLS5_OIEC_CH0_OFFSET = 0, +	HDMI_FC_AUDSCHNLS5_OIEC_CH0_MASK = 0x0f, +	HDMI_FC_AUDSCHNLS5_OIEC_CH1_OFFSET = 4, +	HDMI_FC_AUDSCHNLS5_OIEC_CH1_MASK = 0xf0, +	HDMI_FC_AUDSCHNLS6_OIEC_CH2_OFFSET = 0, +	HDMI_FC_AUDSCHNLS6_OIEC_CH2_MASK = 0x0f, +	HDMI_FC_AUDSCHNLS6_OIEC_CH3_OFFSET = 4, +	HDMI_FC_AUDSCHNLS6_OIEC_CH3_MASK = 0xf0, + +/* HDMI_FC_AUDSCHNLS7 field values */ +	HDMI_FC_AUDSCHNLS7_ACCURACY_OFFSET = 4, +	HDMI_FC_AUDSCHNLS7_ACCURACY_MASK = 0x30, + +/* HDMI_FC_AUDSCHNLS8 field values */ +	HDMI_FC_AUDSCHNLS8_ORIGSAMPFREQ_MASK = 0xf0, +	HDMI_FC_AUDSCHNLS8_ORIGSAMPFREQ_OFFSET = 4, +	HDMI_FC_AUDSCHNLS8_WORDLEGNTH_MASK = 0x0f, +	HDMI_FC_AUDSCHNLS8_WORDLEGNTH_OFFSET = 0, + +/* FC_AUDSCONF field values */ +	HDMI_FC_AUDSCONF_AUD_PACKET_SAMPFIT_MASK = 0xF0, +	HDMI_FC_AUDSCONF_AUD_PACKET_SAMPFIT_OFFSET = 4, +	HDMI_FC_AUDSCONF_AUD_PACKET_LAYOUT_MASK = 0x1, +	HDMI_FC_AUDSCONF_AUD_PACKET_LAYOUT_OFFSET = 0, +	HDMI_FC_AUDSCONF_AUD_PACKET_LAYOUT_LAYOUT1 = 0x1, +	HDMI_FC_AUDSCONF_AUD_PACKET_LAYOUT_LAYOUT0 = 0x0, + +/* FC_STAT2 field values */ +	HDMI_FC_STAT2_OVERFLOW_MASK = 0x03, +	HDMI_FC_STAT2_LOW_PRIORITY_OVERFLOW = 0x02, +	HDMI_FC_STAT2_HIGH_PRIORITY_OVERFLOW = 0x01, + +/* FC_INT2 field values */ +	HDMI_FC_INT2_OVERFLOW_MASK = 0x03, +	HDMI_FC_INT2_LOW_PRIORITY_OVERFLOW = 0x02, +	HDMI_FC_INT2_HIGH_PRIORITY_OVERFLOW = 0x01, + +/* FC_MASK2 field values */ +	HDMI_FC_MASK2_OVERFLOW_MASK = 0x03, +	HDMI_FC_MASK2_LOW_PRIORITY_OVERFLOW = 0x02, +	HDMI_FC_MASK2_HIGH_PRIORITY_OVERFLOW = 0x01, + +/* FC_PRCONF field values */ +	HDMI_FC_PRCONF_INCOMING_PR_FACTOR_MASK = 0xF0, +	HDMI_FC_PRCONF_INCOMING_PR_FACTOR_OFFSET = 4, +	HDMI_FC_PRCONF_OUTPUT_PR_FACTOR_MASK = 0x0F, +	HDMI_FC_PRCONF_OUTPUT_PR_FACTOR_OFFSET = 0, + +/* FC_AVICONF0-FC_AVICONF3 field values */ +	HDMI_FC_AVICONF0_PIX_FMT_MASK = 0x03, +	HDMI_FC_AVICONF0_PIX_FMT_RGB = 0x00, +	HDMI_FC_AVICONF0_PIX_FMT_YCBCR422 = 0x01, +	HDMI_FC_AVICONF0_PIX_FMT_YCBCR444 = 0x02, +	HDMI_FC_AVICONF0_ACTIVE_FMT_MASK = 0x40, +	HDMI_FC_AVICONF0_ACTIVE_FMT_INFO_PRESENT = 0x40, +	HDMI_FC_AVICONF0_ACTIVE_FMT_NO_INFO = 0x00, +	HDMI_FC_AVICONF0_BAR_DATA_MASK = 0x0C, +	HDMI_FC_AVICONF0_BAR_DATA_NO_DATA = 0x00, +	HDMI_FC_AVICONF0_BAR_DATA_VERT_BAR = 0x04, +	HDMI_FC_AVICONF0_BAR_DATA_HORIZ_BAR = 0x08, +	HDMI_FC_AVICONF0_BAR_DATA_VERT_HORIZ_BAR = 0x0C, +	HDMI_FC_AVICONF0_SCAN_INFO_MASK = 0x30, +	HDMI_FC_AVICONF0_SCAN_INFO_OVERSCAN = 0x10, +	HDMI_FC_AVICONF0_SCAN_INFO_UNDERSCAN = 0x20, +	HDMI_FC_AVICONF0_SCAN_INFO_NODATA = 0x00, + +	HDMI_FC_AVICONF1_ACTIVE_ASPECT_RATIO_MASK = 0x0F, +	HDMI_FC_AVICONF1_ACTIVE_ASPECT_RATIO_USE_CODED = 0x08, +	HDMI_FC_AVICONF1_ACTIVE_ASPECT_RATIO_4_3 = 0x09, +	HDMI_FC_AVICONF1_ACTIVE_ASPECT_RATIO_16_9 = 0x0A, +	HDMI_FC_AVICONF1_ACTIVE_ASPECT_RATIO_14_9 = 0x0B, +	HDMI_FC_AVICONF1_CODED_ASPECT_RATIO_MASK = 0x30, +	HDMI_FC_AVICONF1_CODED_ASPECT_RATIO_NO_DATA = 0x00, +	HDMI_FC_AVICONF1_CODED_ASPECT_RATIO_4_3 = 0x10, +	HDMI_FC_AVICONF1_CODED_ASPECT_RATIO_16_9 = 0x20, +	HDMI_FC_AVICONF1_COLORIMETRY_MASK = 0xC0, +	HDMI_FC_AVICONF1_COLORIMETRY_NO_DATA = 0x00, +	HDMI_FC_AVICONF1_COLORIMETRY_SMPTE = 0x40, +	HDMI_FC_AVICONF1_COLORIMETRY_ITUR = 0x80, +	HDMI_FC_AVICONF1_COLORIMETRY_EXTENDED_INFO = 0xC0, + +	HDMI_FC_AVICONF2_SCALING_MASK = 0x03, +	HDMI_FC_AVICONF2_SCALING_NONE = 0x00, +	HDMI_FC_AVICONF2_SCALING_HORIZ = 0x01, +	HDMI_FC_AVICONF2_SCALING_VERT = 0x02, +	HDMI_FC_AVICONF2_SCALING_HORIZ_VERT = 0x03, +	HDMI_FC_AVICONF2_RGB_QUANT_MASK = 0x0C, +	HDMI_FC_AVICONF2_RGB_QUANT_DEFAULT = 0x00, +	HDMI_FC_AVICONF2_RGB_QUANT_LIMITED_RANGE = 0x04, +	HDMI_FC_AVICONF2_RGB_QUANT_FULL_RANGE = 0x08, +	HDMI_FC_AVICONF2_EXT_COLORIMETRY_MASK = 0x70, +	HDMI_FC_AVICONF2_EXT_COLORIMETRY_XVYCC601 = 0x00, +	HDMI_FC_AVICONF2_EXT_COLORIMETRY_XVYCC709 = 0x10, +	HDMI_FC_AVICONF2_EXT_COLORIMETRY_SYCC601 = 0x20, +	HDMI_FC_AVICONF2_EXT_COLORIMETRY_ADOBE_YCC601 = 0x30, +	HDMI_FC_AVICONF2_EXT_COLORIMETRY_ADOBE_RGB = 0x40, +	HDMI_FC_AVICONF2_IT_CONTENT_MASK = 0x80, +	HDMI_FC_AVICONF2_IT_CONTENT_NO_DATA = 0x00, +	HDMI_FC_AVICONF2_IT_CONTENT_VALID = 0x80, + +	HDMI_FC_AVICONF3_IT_CONTENT_TYPE_MASK = 0x03, +	HDMI_FC_AVICONF3_IT_CONTENT_TYPE_GRAPHICS = 0x00, +	HDMI_FC_AVICONF3_IT_CONTENT_TYPE_PHOTO = 0x01, +	HDMI_FC_AVICONF3_IT_CONTENT_TYPE_CINEMA = 0x02, +	HDMI_FC_AVICONF3_IT_CONTENT_TYPE_GAME = 0x03, +	HDMI_FC_AVICONF3_QUANT_RANGE_MASK = 0x0C, +	HDMI_FC_AVICONF3_QUANT_RANGE_LIMITED = 0x00, +	HDMI_FC_AVICONF3_QUANT_RANGE_FULL = 0x04, + +/* FC_DBGFORCE field values */ +	HDMI_FC_DBGFORCE_FORCEAUDIO = 0x10, +	HDMI_FC_DBGFORCE_FORCEVIDEO = 0x1, + +/* PHY_CONF0 field values */ +	HDMI_PHY_CONF0_PDZ_MASK = 0x80, +	HDMI_PHY_CONF0_PDZ_OFFSET = 7, +	HDMI_PHY_CONF0_ENTMDS_MASK = 0x40, +	HDMI_PHY_CONF0_ENTMDS_OFFSET = 6, +	HDMI_PHY_CONF0_SPARECTRL = 0x20, +	HDMI_PHY_CONF0_GEN2_PDDQ_MASK = 0x10, +	HDMI_PHY_CONF0_GEN2_PDDQ_OFFSET = 4, +	HDMI_PHY_CONF0_GEN2_TXPWRON_MASK = 0x8, +	HDMI_PHY_CONF0_GEN2_TXPWRON_OFFSET = 3, +	HDMI_PHY_CONF0_GEN2_ENHPDRXSENSE_MASK = 0x4, +	HDMI_PHY_CONF0_GEN2_ENHPDRXSENSE_OFFSET = 2, +	HDMI_PHY_CONF0_SELDATAENPOL_MASK = 0x2, +	HDMI_PHY_CONF0_SELDATAENPOL_OFFSET = 1, +	HDMI_PHY_CONF0_SELDIPIF_MASK = 0x1, +	HDMI_PHY_CONF0_SELDIPIF_OFFSET = 0, + +/* PHY_TST0 field values */ +	HDMI_PHY_TST0_TSTCLR_MASK = 0x20, +	HDMI_PHY_TST0_TSTCLR_OFFSET = 5, +	HDMI_PHY_TST0_TSTEN_MASK = 0x10, +	HDMI_PHY_TST0_TSTEN_OFFSET = 4, +	HDMI_PHY_TST0_TSTCLK_MASK = 0x1, +	HDMI_PHY_TST0_TSTCLK_OFFSET = 0, + +/* PHY_STAT0 field values */ +	HDMI_PHY_RX_SENSE3 = 0x80, +	HDMI_PHY_RX_SENSE2 = 0x40, +	HDMI_PHY_RX_SENSE1 = 0x20, +	HDMI_PHY_RX_SENSE0 = 0x10, +	HDMI_PHY_HPD = 0x02, +	HDMI_PHY_TX_PHY_LOCK = 0x01, + +/* PHY_I2CM_SLAVE_ADDR field values */ +	HDMI_PHY_I2CM_SLAVE_ADDR_PHY_GEN2 = 0x69, +	HDMI_PHY_I2CM_SLAVE_ADDR_HEAC_PHY = 0x49, + +/* PHY_I2CM_OPERATION_ADDR field values */ +	HDMI_PHY_I2CM_OPERATION_ADDR_WRITE = 0x10, +	HDMI_PHY_I2CM_OPERATION_ADDR_READ = 0x1, + +/* HDMI_PHY_I2CM_INT_ADDR */ +	HDMI_PHY_I2CM_INT_ADDR_DONE_POL = 0x08, +	HDMI_PHY_I2CM_INT_ADDR_DONE_MASK = 0x04, + +/* HDMI_PHY_I2CM_CTLINT_ADDR */ +	HDMI_PHY_I2CM_CTLINT_ADDR_NAC_POL = 0x80, +	HDMI_PHY_I2CM_CTLINT_ADDR_NAC_MASK = 0x40, +	HDMI_PHY_I2CM_CTLINT_ADDR_ARBITRATION_POL = 0x08, +	HDMI_PHY_I2CM_CTLINT_ADDR_ARBITRATION_MASK = 0x04, + +/* AUD_CTS3 field values */ +	HDMI_AUD_CTS3_N_SHIFT_OFFSET = 5, +	HDMI_AUD_CTS3_N_SHIFT_MASK = 0xe0, +	HDMI_AUD_CTS3_N_SHIFT_1 = 0, +	HDMI_AUD_CTS3_N_SHIFT_16 = 0x20, +	HDMI_AUD_CTS3_N_SHIFT_32 = 0x40, +	HDMI_AUD_CTS3_N_SHIFT_64 = 0x60, +	HDMI_AUD_CTS3_N_SHIFT_128 = 0x80, +	HDMI_AUD_CTS3_N_SHIFT_256 = 0xa0, +	/* note that the CTS3 MANUAL bit has been removed +	   from our part. Can't set it, will read as 0. */ +	HDMI_AUD_CTS3_CTS_MANUAL = 0x10, +	HDMI_AUD_CTS3_AUDCTS19_16_MASK = 0x0f, + +/* AHB_DMA_CONF0 field values */ +	HDMI_AHB_DMA_CONF0_SW_FIFO_RST_OFFSET = 7, +	HDMI_AHB_DMA_CONF0_SW_FIFO_RST_MASK = 0x80, +	HDMI_AHB_DMA_CONF0_HBR = 0x10, +	HDMI_AHB_DMA_CONF0_EN_HLOCK_OFFSET = 3, +	HDMI_AHB_DMA_CONF0_EN_HLOCK_MASK = 0x08, +	HDMI_AHB_DMA_CONF0_INCR_TYPE_OFFSET = 1, +	HDMI_AHB_DMA_CONF0_INCR_TYPE_MASK = 0x06, +	HDMI_AHB_DMA_CONF0_INCR4 = 0x0, +	HDMI_AHB_DMA_CONF0_INCR8 = 0x2, +	HDMI_AHB_DMA_CONF0_INCR16 = 0x4, +	HDMI_AHB_DMA_CONF0_BURST_MODE = 0x1, + +/* HDMI_AHB_DMA_START field values */ +	HDMI_AHB_DMA_START_START_OFFSET = 0, +	HDMI_AHB_DMA_START_START_MASK = 0x01, + +/* HDMI_AHB_DMA_STOP field values */ +	HDMI_AHB_DMA_STOP_STOP_OFFSET = 0, +	HDMI_AHB_DMA_STOP_STOP_MASK = 0x01, + +/* AHB_DMA_STAT, AHB_DMA_INT, AHB_DMA_MASK, AHB_DMA_POL field values */ +	HDMI_AHB_DMA_DONE = 0x80, +	HDMI_AHB_DMA_RETRY_SPLIT = 0x40, +	HDMI_AHB_DMA_LOSTOWNERSHIP = 0x20, +	HDMI_AHB_DMA_ERROR = 0x10, +	HDMI_AHB_DMA_FIFO_THREMPTY = 0x04, +	HDMI_AHB_DMA_FIFO_FULL = 0x02, +	HDMI_AHB_DMA_FIFO_EMPTY = 0x01, + +/* AHB_DMA_BUFFSTAT, AHB_DMA_BUFFINT,AHB_DMA_BUFFMASK,AHB_DMA_BUFFPOL values */ +	HDMI_AHB_DMA_BUFFSTAT_FULL = 0x02, +	HDMI_AHB_DMA_BUFFSTAT_EMPTY = 0x01, + +/* MC_CLKDIS field values */ +	HDMI_MC_CLKDIS_HDCPCLK_DISABLE = 0x40, +	HDMI_MC_CLKDIS_CECCLK_DISABLE = 0x20, +	HDMI_MC_CLKDIS_CSCCLK_DISABLE = 0x10, +	HDMI_MC_CLKDIS_AUDCLK_DISABLE = 0x8, +	HDMI_MC_CLKDIS_PREPCLK_DISABLE = 0x4, +	HDMI_MC_CLKDIS_TMDSCLK_DISABLE = 0x2, +	HDMI_MC_CLKDIS_PIXELCLK_DISABLE = 0x1, + +/* MC_SWRSTZ field values */ +	HDMI_MC_SWRSTZ_TMDSSWRST_REQ = 0x02, + +/* MC_FLOWCTRL field values */ +	HDMI_MC_FLOWCTRL_FEED_THROUGH_OFF_MASK = 0x1, +	HDMI_MC_FLOWCTRL_FEED_THROUGH_OFF_CSC_IN_PATH = 0x1, +	HDMI_MC_FLOWCTRL_FEED_THROUGH_OFF_CSC_BYPASS = 0x0, + +/* MC_PHYRSTZ field values */ +	HDMI_MC_PHYRSTZ_ASSERT = 0x0, +	HDMI_MC_PHYRSTZ_DEASSERT = 0x1, + +/* MC_HEACPHY_RST field values */ +	HDMI_MC_HEACPHY_RST_ASSERT = 0x1, +	HDMI_MC_HEACPHY_RST_DEASSERT = 0x0, + +/* CSC_CFG field values */ +	HDMI_CSC_CFG_INTMODE_MASK = 0x30, +	HDMI_CSC_CFG_INTMODE_OFFSET = 4, +	HDMI_CSC_CFG_INTMODE_DISABLE = 0x00, +	HDMI_CSC_CFG_INTMODE_CHROMA_INT_FORMULA1 = 0x10, +	HDMI_CSC_CFG_INTMODE_CHROMA_INT_FORMULA2 = 0x20, +	HDMI_CSC_CFG_DECMODE_MASK = 0x3, +	HDMI_CSC_CFG_DECMODE_OFFSET = 0, +	HDMI_CSC_CFG_DECMODE_DISABLE = 0x0, +	HDMI_CSC_CFG_DECMODE_CHROMA_INT_FORMULA1 = 0x1, +	HDMI_CSC_CFG_DECMODE_CHROMA_INT_FORMULA2 = 0x2, +	HDMI_CSC_CFG_DECMODE_CHROMA_INT_FORMULA3 = 0x3, + +/* CSC_SCALE field values */ +	HDMI_CSC_SCALE_CSC_COLORDE_PTH_MASK = 0xF0, +	HDMI_CSC_SCALE_CSC_COLORDE_PTH_24BPP = 0x00, +	HDMI_CSC_SCALE_CSC_COLORDE_PTH_30BPP = 0x50, +	HDMI_CSC_SCALE_CSC_COLORDE_PTH_36BPP = 0x60, +	HDMI_CSC_SCALE_CSC_COLORDE_PTH_48BPP = 0x70, +	HDMI_CSC_SCALE_CSCSCALE_MASK = 0x03, + +/* A_HDCPCFG0 field values */ +	HDMI_A_HDCPCFG0_ELVENA_MASK = 0x80, +	HDMI_A_HDCPCFG0_ELVENA_ENABLE = 0x80, +	HDMI_A_HDCPCFG0_ELVENA_DISABLE = 0x00, +	HDMI_A_HDCPCFG0_I2CFASTMODE_MASK = 0x40, +	HDMI_A_HDCPCFG0_I2CFASTMODE_ENABLE = 0x40, +	HDMI_A_HDCPCFG0_I2CFASTMODE_DISABLE = 0x00, +	HDMI_A_HDCPCFG0_BYPENCRYPTION_MASK = 0x20, +	HDMI_A_HDCPCFG0_BYPENCRYPTION_ENABLE = 0x20, +	HDMI_A_HDCPCFG0_BYPENCRYPTION_DISABLE = 0x00, +	HDMI_A_HDCPCFG0_SYNCRICHECK_MASK = 0x10, +	HDMI_A_HDCPCFG0_SYNCRICHECK_ENABLE = 0x10, +	HDMI_A_HDCPCFG0_SYNCRICHECK_DISABLE = 0x00, +	HDMI_A_HDCPCFG0_AVMUTE_MASK = 0x8, +	HDMI_A_HDCPCFG0_AVMUTE_ENABLE = 0x8, +	HDMI_A_HDCPCFG0_AVMUTE_DISABLE = 0x0, +	HDMI_A_HDCPCFG0_RXDETECT_MASK = 0x4, +	HDMI_A_HDCPCFG0_RXDETECT_ENABLE = 0x4, +	HDMI_A_HDCPCFG0_RXDETECT_DISABLE = 0x0, +	HDMI_A_HDCPCFG0_EN11FEATURE_MASK = 0x2, +	HDMI_A_HDCPCFG0_EN11FEATURE_ENABLE = 0x2, +	HDMI_A_HDCPCFG0_EN11FEATURE_DISABLE = 0x0, +	HDMI_A_HDCPCFG0_HDMIDVI_MASK = 0x1, +	HDMI_A_HDCPCFG0_HDMIDVI_HDMI = 0x1, +	HDMI_A_HDCPCFG0_HDMIDVI_DVI = 0x0, + +/* A_HDCPCFG1 field values */ +	HDMI_A_HDCPCFG1_DISSHA1CHECK_MASK = 0x8, +	HDMI_A_HDCPCFG1_DISSHA1CHECK_DISABLE = 0x8, +	HDMI_A_HDCPCFG1_DISSHA1CHECK_ENABLE = 0x0, +	HDMI_A_HDCPCFG1_PH2UPSHFTENC_MASK = 0x4, +	HDMI_A_HDCPCFG1_PH2UPSHFTENC_ENABLE = 0x4, +	HDMI_A_HDCPCFG1_PH2UPSHFTENC_DISABLE = 0x0, +	HDMI_A_HDCPCFG1_ENCRYPTIONDISABLE_MASK = 0x2, +	HDMI_A_HDCPCFG1_ENCRYPTIONDISABLE_DISABLE = 0x2, +	HDMI_A_HDCPCFG1_ENCRYPTIONDISABLE_ENABLE = 0x0, +	HDMI_A_HDCPCFG1_SWRESET_MASK = 0x1, +	HDMI_A_HDCPCFG1_SWRESET_ASSERT = 0x0, + +/* A_VIDPOLCFG field values */ +	HDMI_A_VIDPOLCFG_UNENCRYPTCONF_MASK = 0x60, +	HDMI_A_VIDPOLCFG_UNENCRYPTCONF_OFFSET = 5, +	HDMI_A_VIDPOLCFG_DATAENPOL_MASK = 0x10, +	HDMI_A_VIDPOLCFG_DATAENPOL_ACTIVE_HIGH = 0x10, +	HDMI_A_VIDPOLCFG_DATAENPOL_ACTIVE_LOW = 0x0, +	HDMI_A_VIDPOLCFG_VSYNCPOL_MASK = 0x8, +	HDMI_A_VIDPOLCFG_VSYNCPOL_ACTIVE_HIGH = 0x8, +	HDMI_A_VIDPOLCFG_VSYNCPOL_ACTIVE_LOW = 0x0, +	HDMI_A_VIDPOLCFG_HSYNCPOL_MASK = 0x2, +	HDMI_A_VIDPOLCFG_HSYNCPOL_ACTIVE_HIGH = 0x2, +	HDMI_A_VIDPOLCFG_HSYNCPOL_ACTIVE_LOW = 0x0, +}; +#endif /* __IMX_HDMI_H__ */ diff --git a/drivers/staging/imx-drm/imx-ldb.c b/drivers/staging/imx-drm/imx-ldb.c index af733ea4856..7e3f019d7e7 100644 --- a/drivers/staging/imx-drm/imx-ldb.c +++ b/drivers/staging/imx-drm/imx-ldb.c @@ -20,6 +20,7 @@  #include <linux/module.h>  #include <linux/clk.h> +#include <linux/component.h>  #include <drm/drmP.h>  #include <drm/drm_fb_helper.h>  #include <drm/drm_crtc_helper.h> @@ -58,9 +59,8 @@ struct imx_ldb;  struct imx_ldb_channel {  	struct imx_ldb *ldb;  	struct drm_connector connector; -	struct imx_drm_connector *imx_drm_connector;  	struct drm_encoder encoder; -	struct imx_drm_encoder *imx_drm_encoder; +	struct device_node *child;  	int chno;  	void *edid;  	int edid_len; @@ -91,11 +91,6 @@ static enum drm_connector_status imx_ldb_connector_detect(  	return connector_status_connected;  } -static void imx_ldb_connector_destroy(struct drm_connector *connector) -{ -	/* do not free here */ -} -  static int imx_ldb_connector_get_modes(struct drm_connector *connector)  {  	struct imx_ldb_channel *imx_ldb_ch = con_to_imx_ldb_ch(connector); @@ -111,6 +106,8 @@ static int imx_ldb_connector_get_modes(struct drm_connector *connector)  		struct drm_display_mode *mode;  		mode = drm_mode_create(connector->dev); +		if (!mode) +			return -EINVAL;  		drm_mode_copy(mode, &imx_ldb_ch->mode);  		mode->type |= DRM_MODE_TYPE_DRIVER | DRM_MODE_TYPE_PREFERRED;  		drm_mode_probed_add(connector, mode); @@ -120,12 +117,6 @@ static int imx_ldb_connector_get_modes(struct drm_connector *connector)  	return num_modes;  } -static int imx_ldb_connector_mode_valid(struct drm_connector *connector, -			  struct drm_display_mode *mode) -{ -	return 0; -} -  static struct drm_encoder *imx_ldb_connector_best_encoder(  		struct drm_connector *connector)  { @@ -167,9 +158,10 @@ static void imx_ldb_set_clock(struct imx_ldb *ldb, int mux, int chno,  	/* set display clock mux to LDB input clock */  	ret = clk_set_parent(ldb->clk_sel[mux], ldb->clk[chno]); -	if (ret) { -		dev_err(ldb->dev, "unable to set di%d parent clock to ldb_di%d\n", mux, chno); -	} +	if (ret) +		dev_err(ldb->dev, +			"unable to set di%d parent clock to ldb_di%d\n", mux, +			chno);  }  static void imx_ldb_encoder_prepare(struct drm_encoder *encoder) @@ -180,8 +172,7 @@ static void imx_ldb_encoder_prepare(struct drm_encoder *encoder)  	u32 pixel_fmt;  	unsigned long serial_clk;  	unsigned long di_clk = mode->clock * 1000; -	int mux = imx_drm_encoder_get_mux_id(imx_ldb_ch->imx_drm_encoder, -					     encoder->crtc); +	int mux = imx_drm_encoder_get_mux_id(imx_ldb_ch->child, encoder);  	if (ldb->ldb_ctrl & LDB_SPLIT_MODE_EN) {  		/* dual channel LVDS mode */ @@ -190,7 +181,8 @@ static void imx_ldb_encoder_prepare(struct drm_encoder *encoder)  		imx_ldb_set_clock(ldb, mux, 1, serial_clk, di_clk);  	} else {  		serial_clk = 7000UL * mode->clock; -		imx_ldb_set_clock(ldb, mux, imx_ldb_ch->chno, serial_clk, di_clk); +		imx_ldb_set_clock(ldb, mux, imx_ldb_ch->chno, serial_clk, +				di_clk);  	}  	switch (imx_ldb_ch->chno) { @@ -208,8 +200,7 @@ static void imx_ldb_encoder_prepare(struct drm_encoder *encoder)  		pixel_fmt = V4L2_PIX_FMT_RGB24;  	} -	imx_drm_crtc_panel_format(encoder->crtc, DRM_MODE_ENCODER_LVDS, -			pixel_fmt); +	imx_drm_panel_format(encoder, pixel_fmt);  }  static void imx_ldb_encoder_commit(struct drm_encoder *encoder) @@ -217,8 +208,7 @@ static void imx_ldb_encoder_commit(struct drm_encoder *encoder)  	struct imx_ldb_channel *imx_ldb_ch = enc_to_imx_ldb_ch(encoder);  	struct imx_ldb *ldb = imx_ldb_ch->ldb;  	int dual = ldb->ldb_ctrl & LDB_SPLIT_MODE_EN; -	int mux = imx_drm_encoder_get_mux_id(imx_ldb_ch->imx_drm_encoder, -					     encoder->crtc); +	int mux = imx_drm_encoder_get_mux_id(imx_ldb_ch->child, encoder);  	if (dual) {  		clk_prepare_enable(ldb->clk[0]); @@ -317,26 +307,20 @@ static void imx_ldb_encoder_disable(struct drm_encoder *encoder)  	}  } -static void imx_ldb_encoder_destroy(struct drm_encoder *encoder) -{ -	/* do not free here */ -} -  static struct drm_connector_funcs imx_ldb_connector_funcs = {  	.dpms = drm_helper_connector_dpms,  	.fill_modes = drm_helper_probe_single_connector_modes,  	.detect = imx_ldb_connector_detect, -	.destroy = imx_ldb_connector_destroy, +	.destroy = imx_drm_connector_destroy,  };  static struct drm_connector_helper_funcs imx_ldb_connector_helper_funcs = {  	.get_modes = imx_ldb_connector_get_modes,  	.best_encoder = imx_ldb_connector_best_encoder, -	.mode_valid = imx_ldb_connector_mode_valid,  };  static struct drm_encoder_funcs imx_ldb_encoder_funcs = { -	.destroy = imx_ldb_encoder_destroy, +	.destroy = imx_drm_encoder_destroy,  };  static struct drm_encoder_helper_funcs imx_ldb_encoder_helper_funcs = { @@ -352,58 +336,47 @@ static int imx_ldb_get_clk(struct imx_ldb *ldb, int chno)  {  	char clkname[16]; -	sprintf(clkname, "di%d", chno); +	snprintf(clkname, sizeof(clkname), "di%d", chno);  	ldb->clk[chno] = devm_clk_get(ldb->dev, clkname);  	if (IS_ERR(ldb->clk[chno]))  		return PTR_ERR(ldb->clk[chno]); -	sprintf(clkname, "di%d_pll", chno); +	snprintf(clkname, sizeof(clkname), "di%d_pll", chno);  	ldb->clk_pll[chno] = devm_clk_get(ldb->dev, clkname); -	if (IS_ERR(ldb->clk_pll[chno])) -		return PTR_ERR(ldb->clk_pll[chno]); -	return 0; +	return PTR_ERR_OR_ZERO(ldb->clk_pll[chno]);  } -static int imx_ldb_register(struct imx_ldb_channel *imx_ldb_ch) +static int imx_ldb_register(struct drm_device *drm, +	struct imx_ldb_channel *imx_ldb_ch)  { -	int ret;  	struct imx_ldb *ldb = imx_ldb_ch->ldb; +	int ret; + +	ret = imx_drm_encoder_parse_of(drm, &imx_ldb_ch->encoder, +				       imx_ldb_ch->child); +	if (ret) +		return ret;  	ret = imx_ldb_get_clk(ldb, imx_ldb_ch->chno);  	if (ret)  		return ret; +  	if (ldb->ldb_ctrl & LDB_SPLIT_MODE_EN) { -		ret |= imx_ldb_get_clk(ldb, 1); +		ret = imx_ldb_get_clk(ldb, 1);  		if (ret)  			return ret;  	} -	imx_ldb_ch->connector.funcs = &imx_ldb_connector_funcs; -	imx_ldb_ch->encoder.funcs = &imx_ldb_encoder_funcs; - -	imx_ldb_ch->encoder.encoder_type = DRM_MODE_ENCODER_LVDS; -	imx_ldb_ch->connector.connector_type = DRM_MODE_CONNECTOR_LVDS; -  	drm_encoder_helper_add(&imx_ldb_ch->encoder,  			&imx_ldb_encoder_helper_funcs); -	ret = imx_drm_add_encoder(&imx_ldb_ch->encoder, -			&imx_ldb_ch->imx_drm_encoder, THIS_MODULE); -	if (ret) { -		dev_err(ldb->dev, "adding encoder failed with %d\n", ret); -		return ret; -	} +	drm_encoder_init(drm, &imx_ldb_ch->encoder, &imx_ldb_encoder_funcs, +			 DRM_MODE_ENCODER_LVDS);  	drm_connector_helper_add(&imx_ldb_ch->connector,  			&imx_ldb_connector_helper_funcs); - -	ret = imx_drm_add_connector(&imx_ldb_ch->connector, -			&imx_ldb_ch->imx_drm_connector, THIS_MODULE); -	if (ret) { -		imx_drm_remove_encoder(imx_ldb_ch->imx_drm_encoder); -		dev_err(ldb->dev, "adding connector failed with %d\n", ret); -		return ret; -	} +	drm_connector_init(drm, &imx_ldb_ch->connector, +			   &imx_ldb_connector_funcs, DRM_MODE_CONNECTOR_LVDS);  	drm_mode_connector_attach_encoder(&imx_ldb_ch->connector,  			&imx_ldb_ch->encoder); @@ -416,12 +389,12 @@ enum {  	LVDS_BIT_MAP_JEIDA  }; -static const char *imx_ldb_bit_mappings[] = { +static const char * const imx_ldb_bit_mappings[] = {  	[LVDS_BIT_MAP_SPWG]  = "spwg",  	[LVDS_BIT_MAP_JEIDA] = "jeida",  }; -const int of_get_data_mapping(struct device_node *np) +static const int of_get_data_mapping(struct device_node *np)  {  	const char *bm;  	int ret, i; @@ -462,12 +435,12 @@ static const struct of_device_id imx_ldb_dt_ids[] = {  };  MODULE_DEVICE_TABLE(of, imx_ldb_dt_ids); -static int imx_ldb_probe(struct platform_device *pdev) +static int imx_ldb_bind(struct device *dev, struct device *master, void *data)  { -	struct device_node *np = pdev->dev.of_node; +	struct drm_device *drm = data; +	struct device_node *np = dev->of_node;  	const struct of_device_id *of_id = -			of_match_device(of_match_ptr(imx_ldb_dt_ids), -					&pdev->dev); +			of_match_device(imx_ldb_dt_ids, dev);  	struct device_node *child;  	const u8 *edidp;  	struct imx_ldb *imx_ldb; @@ -477,17 +450,17 @@ static int imx_ldb_probe(struct platform_device *pdev)  	int ret;  	int i; -	imx_ldb = devm_kzalloc(&pdev->dev, sizeof(*imx_ldb), GFP_KERNEL); +	imx_ldb = devm_kzalloc(dev, sizeof(*imx_ldb), GFP_KERNEL);  	if (!imx_ldb)  		return -ENOMEM;  	imx_ldb->regmap = syscon_regmap_lookup_by_phandle(np, "gpr");  	if (IS_ERR(imx_ldb->regmap)) { -		dev_err(&pdev->dev, "failed to get parent regmap\n"); +		dev_err(dev, "failed to get parent regmap\n");  		return PTR_ERR(imx_ldb->regmap);  	} -	imx_ldb->dev = &pdev->dev; +	imx_ldb->dev = dev;  	if (of_id)  		imx_ldb->lvds_mux = of_id->data; @@ -525,7 +498,7 @@ static int imx_ldb_probe(struct platform_device *pdev)  			return -EINVAL;  		if (dual && i > 0) { -			dev_warn(&pdev->dev, "dual-channel mode, ignoring second output\n"); +			dev_warn(dev, "dual-channel mode, ignoring second output\n");  			continue;  		} @@ -535,6 +508,7 @@ static int imx_ldb_probe(struct platform_device *pdev)  		channel = &imx_ldb->channel[i];  		channel->ldb = imx_ldb;  		channel->chno = i; +		channel->child = child;  		edidp = of_get_property(child, "edid", &channel->edid_len);  		if (edidp) { @@ -557,54 +531,67 @@ static int imx_ldb_probe(struct platform_device *pdev)  		case LVDS_BIT_MAP_SPWG:  			if (datawidth == 24) {  				if (i == 0 || dual) -					imx_ldb->ldb_ctrl |= LDB_DATA_WIDTH_CH0_24; +					imx_ldb->ldb_ctrl |= +						LDB_DATA_WIDTH_CH0_24;  				if (i == 1 || dual) -					imx_ldb->ldb_ctrl |= LDB_DATA_WIDTH_CH1_24; +					imx_ldb->ldb_ctrl |= +						LDB_DATA_WIDTH_CH1_24;  			}  			break;  		case LVDS_BIT_MAP_JEIDA:  			if (datawidth == 18) { -				dev_err(&pdev->dev, "JEIDA standard only supported in 24 bit\n"); +				dev_err(dev, "JEIDA standard only supported in 24 bit\n");  				return -EINVAL;  			}  			if (i == 0 || dual) -				imx_ldb->ldb_ctrl |= LDB_DATA_WIDTH_CH0_24 | LDB_BIT_MAP_CH0_JEIDA; +				imx_ldb->ldb_ctrl |= LDB_DATA_WIDTH_CH0_24 | +					LDB_BIT_MAP_CH0_JEIDA;  			if (i == 1 || dual) -				imx_ldb->ldb_ctrl |= LDB_DATA_WIDTH_CH1_24 | LDB_BIT_MAP_CH1_JEIDA; +				imx_ldb->ldb_ctrl |= LDB_DATA_WIDTH_CH1_24 | +					LDB_BIT_MAP_CH1_JEIDA;  			break;  		default: -			dev_err(&pdev->dev, "data mapping not specified or invalid\n"); +			dev_err(dev, "data mapping not specified or invalid\n");  			return -EINVAL;  		} -		ret = imx_ldb_register(channel); +		ret = imx_ldb_register(drm, channel);  		if (ret)  			return ret; - -		imx_drm_encoder_add_possible_crtcs(channel->imx_drm_encoder, child);  	} -	platform_set_drvdata(pdev, imx_ldb); +	dev_set_drvdata(dev, imx_ldb);  	return 0;  } -static int imx_ldb_remove(struct platform_device *pdev) +static void imx_ldb_unbind(struct device *dev, struct device *master, +	void *data)  { -	struct imx_ldb *imx_ldb = platform_get_drvdata(pdev); +	struct imx_ldb *imx_ldb = dev_get_drvdata(dev);  	int i;  	for (i = 0; i < 2; i++) {  		struct imx_ldb_channel *channel = &imx_ldb->channel[i]; -		struct drm_connector *connector = &channel->connector; -		struct drm_encoder *encoder = &channel->encoder; -		drm_mode_connector_detach_encoder(connector, encoder); - -		imx_drm_remove_connector(channel->imx_drm_connector); -		imx_drm_remove_encoder(channel->imx_drm_encoder); +		channel->connector.funcs->destroy(&channel->connector); +		channel->encoder.funcs->destroy(&channel->encoder);  	} +} + +static const struct component_ops imx_ldb_ops = { +	.bind	= imx_ldb_bind, +	.unbind	= imx_ldb_unbind, +}; + +static int imx_ldb_probe(struct platform_device *pdev) +{ +	return component_add(&pdev->dev, &imx_ldb_ops); +} +static int imx_ldb_remove(struct platform_device *pdev) +{ +	component_del(&pdev->dev, &imx_ldb_ops);  	return 0;  } diff --git a/drivers/staging/imx-drm/imx-tve.c b/drivers/staging/imx-drm/imx-tve.c index 33d6525cf99..c628fcdc22a 100644 --- a/drivers/staging/imx-drm/imx-tve.c +++ b/drivers/staging/imx-drm/imx-tve.c @@ -20,6 +20,7 @@  #include <linux/clk.h>  #include <linux/clk-provider.h> +#include <linux/component.h>  #include <linux/module.h>  #include <linux/i2c.h>  #include <linux/regmap.h> @@ -29,6 +30,7 @@  #include <drm/drmP.h>  #include <drm/drm_fb_helper.h>  #include <drm/drm_crtc_helper.h> +#include <video/imx-ipu-v3.h>  #include "imx-drm.h" @@ -110,11 +112,8 @@ enum {  struct imx_tve {  	struct drm_connector connector; -	struct imx_drm_connector *imx_drm_connector;  	struct drm_encoder encoder; -	struct imx_drm_encoder *imx_drm_encoder;  	struct device *dev; -	spinlock_t enable_lock;	/* serializes tve_enable/disable */  	spinlock_t lock;	/* register lock */  	bool enabled;  	int mode; @@ -146,12 +145,10 @@ __releases(&tve->lock)  static void tve_enable(struct imx_tve *tve)  { -	unsigned long flags;  	int ret; -	spin_lock_irqsave(&tve->enable_lock, flags);  	if (!tve->enabled) { -		tve->enabled = 1; +		tve->enabled = true;  		clk_prepare_enable(tve->clk);  		ret = regmap_update_bits(tve->regmap, TVE_COM_CONF_REG,  					 TVE_IPU_CLK_EN | TVE_EN, @@ -169,23 +166,18 @@ static void tve_enable(struct imx_tve *tve)  			     TVE_CD_SM_IEN |  			     TVE_CD_LM_IEN |  			     TVE_CD_MON_END_IEN); - -	spin_unlock_irqrestore(&tve->enable_lock, flags);  }  static void tve_disable(struct imx_tve *tve)  { -	unsigned long flags;  	int ret; -	spin_lock_irqsave(&tve->enable_lock, flags);  	if (tve->enabled) { -		tve->enabled = 0; +		tve->enabled = false;  		ret = regmap_update_bits(tve->regmap, TVE_COM_CONF_REG,  					 TVE_IPU_CLK_EN | TVE_EN, 0);  		clk_disable_unprepare(tve->clk);  	} -	spin_unlock_irqrestore(&tve->enable_lock, flags);  }  static int tve_setup_tvout(struct imx_tve *tve) @@ -233,11 +225,6 @@ static enum drm_connector_status imx_tve_connector_detect(  	return connector_status_connected;  } -static void imx_tve_connector_destroy(struct drm_connector *connector) -{ -	/* do not free here */ -} -  static int imx_tve_connector_get_modes(struct drm_connector *connector)  {  	struct imx_tve *tve = con_to_tve(connector); @@ -313,13 +300,11 @@ static void imx_tve_encoder_prepare(struct drm_encoder *encoder)  	switch (tve->mode) {  	case TVE_MODE_VGA: -		imx_drm_crtc_panel_format_pins(encoder->crtc, -				DRM_MODE_ENCODER_DAC, IPU_PIX_FMT_GBR24, +		imx_drm_panel_format_pins(encoder, IPU_PIX_FMT_GBR24,  				tve->hsync_pin, tve->vsync_pin);  		break;  	case TVE_MODE_TVOUT: -		imx_drm_crtc_panel_format(encoder->crtc, DRM_MODE_ENCODER_TVDAC, -					  V4L2_PIX_FMT_YUV444); +		imx_drm_panel_format(encoder, V4L2_PIX_FMT_YUV444);  		break;  	}  } @@ -372,16 +357,11 @@ static void imx_tve_encoder_disable(struct drm_encoder *encoder)  	tve_disable(tve);  } -static void imx_tve_encoder_destroy(struct drm_encoder *encoder) -{ -	/* do not free here */ -} -  static struct drm_connector_funcs imx_tve_connector_funcs = {  	.dpms = drm_helper_connector_dpms,  	.fill_modes = drm_helper_probe_single_connector_modes,  	.detect = imx_tve_connector_detect, -	.destroy = imx_tve_connector_destroy, +	.destroy = imx_drm_connector_destroy,  };  static struct drm_connector_helper_funcs imx_tve_connector_helper_funcs = { @@ -391,7 +371,7 @@ static struct drm_connector_helper_funcs imx_tve_connector_helper_funcs = {  };  static struct drm_encoder_funcs imx_tve_encoder_funcs = { -	.destroy = imx_tve_encoder_destroy, +	.destroy = imx_drm_encoder_destroy,  };  static struct drm_encoder_helper_funcs imx_tve_encoder_helper_funcs = { @@ -511,34 +491,27 @@ static int tve_clk_init(struct imx_tve *tve, void __iomem *base)  	return 0;  } -static int imx_tve_register(struct imx_tve *tve) +static int imx_tve_register(struct drm_device *drm, struct imx_tve *tve)  { +	int encoder_type;  	int ret; -	tve->connector.funcs = &imx_tve_connector_funcs; -	tve->encoder.funcs = &imx_tve_encoder_funcs; +	encoder_type = tve->mode == TVE_MODE_VGA ? +				DRM_MODE_ENCODER_DAC : DRM_MODE_ENCODER_TVDAC; -	tve->encoder.encoder_type = DRM_MODE_ENCODER_NONE; -	tve->connector.connector_type = DRM_MODE_CONNECTOR_VGA; +	ret = imx_drm_encoder_parse_of(drm, &tve->encoder, +				       tve->dev->of_node); +	if (ret) +		return ret;  	drm_encoder_helper_add(&tve->encoder, &imx_tve_encoder_helper_funcs); -	ret = imx_drm_add_encoder(&tve->encoder, &tve->imx_drm_encoder, -			THIS_MODULE); -	if (ret) { -		dev_err(tve->dev, "adding encoder failed with %d\n", ret); -		return ret; -	} +	drm_encoder_init(drm, &tve->encoder, &imx_tve_encoder_funcs, +			 encoder_type);  	drm_connector_helper_add(&tve->connector,  			&imx_tve_connector_helper_funcs); - -	ret = imx_drm_add_connector(&tve->connector, -			&tve->imx_drm_connector, THIS_MODULE); -	if (ret) { -		imx_drm_remove_encoder(tve->imx_drm_encoder); -		dev_err(tve->dev, "adding connector failed with %d\n", ret); -		return ret; -	} +	drm_connector_init(drm, &tve->connector, &imx_tve_connector_funcs, +			   DRM_MODE_CONNECTOR_VGA);  	drm_mode_connector_attach_encoder(&tve->connector, &tve->encoder); @@ -568,7 +541,7 @@ static const char *imx_tve_modes[] = {  	[TVE_MODE_VGA] = "vga",  }; -const int of_get_tve_mode(struct device_node *np) +static const int of_get_tve_mode(struct device_node *np)  {  	const char *bm;  	int ret, i; @@ -584,9 +557,11 @@ const int of_get_tve_mode(struct device_node *np)  	return -EINVAL;  } -static int imx_tve_probe(struct platform_device *pdev) +static int imx_tve_bind(struct device *dev, struct device *master, void *data)  { -	struct device_node *np = pdev->dev.of_node; +	struct platform_device *pdev = to_platform_device(dev); +	struct drm_device *drm = data; +	struct device_node *np = dev->of_node;  	struct device_node *ddc_node;  	struct imx_tve *tve;  	struct resource *res; @@ -595,15 +570,14 @@ static int imx_tve_probe(struct platform_device *pdev)  	int irq;  	int ret; -	tve = devm_kzalloc(&pdev->dev, sizeof(*tve), GFP_KERNEL); +	tve = devm_kzalloc(dev, sizeof(*tve), GFP_KERNEL);  	if (!tve)  		return -ENOMEM; -	tve->dev = &pdev->dev; +	tve->dev = dev;  	spin_lock_init(&tve->lock); -	spin_lock_init(&tve->enable_lock); -	ddc_node = of_parse_phandle(np, "ddc", 0); +	ddc_node = of_parse_phandle(np, "ddc-i2c-bus", 0);  	if (ddc_node) {  		tve->ddc = of_find_i2c_adapter_by_node(ddc_node);  		of_node_put(ddc_node); @@ -611,7 +585,7 @@ static int imx_tve_probe(struct platform_device *pdev)  	tve->mode = of_get_tve_mode(np);  	if (tve->mode != TVE_MODE_VGA) { -		dev_err(&pdev->dev, "only VGA mode supported, currently\n"); +		dev_err(dev, "only VGA mode supported, currently\n");  		return -EINVAL;  	} @@ -620,7 +594,7 @@ static int imx_tve_probe(struct platform_device *pdev)  					   &tve->hsync_pin);  		if (ret < 0) { -			dev_err(&pdev->dev, "failed to get vsync pin\n"); +			dev_err(dev, "failed to get vsync pin\n");  			return ret;  		} @@ -628,40 +602,40 @@ static int imx_tve_probe(struct platform_device *pdev)  					    &tve->vsync_pin);  		if (ret < 0) { -			dev_err(&pdev->dev, "failed to get vsync pin\n"); +			dev_err(dev, "failed to get vsync pin\n");  			return ret;  		}  	}  	res = platform_get_resource(pdev, IORESOURCE_MEM, 0); -	base = devm_ioremap_resource(&pdev->dev, res); +	base = devm_ioremap_resource(dev, res);  	if (IS_ERR(base))  		return PTR_ERR(base);  	tve_regmap_config.lock_arg = tve; -	tve->regmap = devm_regmap_init_mmio_clk(&pdev->dev, "tve", base, +	tve->regmap = devm_regmap_init_mmio_clk(dev, "tve", base,  						&tve_regmap_config);  	if (IS_ERR(tve->regmap)) { -		dev_err(&pdev->dev, "failed to init regmap: %ld\n", +		dev_err(dev, "failed to init regmap: %ld\n",  			PTR_ERR(tve->regmap));  		return PTR_ERR(tve->regmap);  	}  	irq = platform_get_irq(pdev, 0);  	if (irq < 0) { -		dev_err(&pdev->dev, "failed to get irq\n"); +		dev_err(dev, "failed to get irq\n");  		return irq;  	} -	ret = devm_request_threaded_irq(&pdev->dev, irq, NULL, +	ret = devm_request_threaded_irq(dev, irq, NULL,  					imx_tve_irq_handler, IRQF_ONESHOT,  					"imx-tve", tve);  	if (ret < 0) { -		dev_err(&pdev->dev, "failed to request irq: %d\n", ret); +		dev_err(dev, "failed to request irq: %d\n", ret);  		return ret;  	} -	tve->dac_reg = devm_regulator_get(&pdev->dev, "dac"); +	tve->dac_reg = devm_regulator_get(dev, "dac");  	if (!IS_ERR(tve->dac_reg)) {  		regulator_set_voltage(tve->dac_reg, 2750000, 2750000);  		ret = regulator_enable(tve->dac_reg); @@ -669,17 +643,17 @@ static int imx_tve_probe(struct platform_device *pdev)  			return ret;  	} -	tve->clk = devm_clk_get(&pdev->dev, "tve"); +	tve->clk = devm_clk_get(dev, "tve");  	if (IS_ERR(tve->clk)) { -		dev_err(&pdev->dev, "failed to get high speed tve clock: %ld\n", +		dev_err(dev, "failed to get high speed tve clock: %ld\n",  			PTR_ERR(tve->clk));  		return PTR_ERR(tve->clk);  	}  	/* this is the IPU DI clock input selector, can be parented to tve_di */ -	tve->di_sel_clk = devm_clk_get(&pdev->dev, "di_sel"); +	tve->di_sel_clk = devm_clk_get(dev, "di_sel");  	if (IS_ERR(tve->di_sel_clk)) { -		dev_err(&pdev->dev, "failed to get ipu di mux clock: %ld\n", +		dev_err(dev, "failed to get ipu di mux clock: %ld\n",  			PTR_ERR(tve->di_sel_clk));  		return PTR_ERR(tve->di_sel_clk);  	} @@ -690,42 +664,51 @@ static int imx_tve_probe(struct platform_device *pdev)  	ret = regmap_read(tve->regmap, TVE_COM_CONF_REG, &val);  	if (ret < 0) { -		dev_err(&pdev->dev, "failed to read configuration register: %d\n", ret); +		dev_err(dev, "failed to read configuration register: %d\n", ret);  		return ret;  	}  	if (val != 0x00100000) { -		dev_err(&pdev->dev, "configuration register default value indicates this is not a TVEv2\n"); +		dev_err(dev, "configuration register default value indicates this is not a TVEv2\n");  		return -ENODEV; -	}; +	}  	/* disable cable detection for VGA mode */  	ret = regmap_write(tve->regmap, TVE_CD_CONT_REG, 0); -	ret = imx_tve_register(tve); +	ret = imx_tve_register(drm, tve);  	if (ret)  		return ret; -	ret = imx_drm_encoder_add_possible_crtcs(tve->imx_drm_encoder, np); - -	platform_set_drvdata(pdev, tve); +	dev_set_drvdata(dev, tve);  	return 0;  } -static int imx_tve_remove(struct platform_device *pdev) +static void imx_tve_unbind(struct device *dev, struct device *master, +	void *data)  { -	struct imx_tve *tve = platform_get_drvdata(pdev); -	struct drm_connector *connector = &tve->connector; -	struct drm_encoder *encoder = &tve->encoder; +	struct imx_tve *tve = dev_get_drvdata(dev); -	drm_mode_connector_detach_encoder(connector, encoder); - -	imx_drm_remove_connector(tve->imx_drm_connector); -	imx_drm_remove_encoder(tve->imx_drm_encoder); +	tve->connector.funcs->destroy(&tve->connector); +	tve->encoder.funcs->destroy(&tve->encoder);  	if (!IS_ERR(tve->dac_reg))  		regulator_disable(tve->dac_reg); +} + +static const struct component_ops imx_tve_ops = { +	.bind	= imx_tve_bind, +	.unbind	= imx_tve_unbind, +}; +static int imx_tve_probe(struct platform_device *pdev) +{ +	return component_add(&pdev->dev, &imx_tve_ops); +} + +static int imx_tve_remove(struct platform_device *pdev) +{ +	component_del(&pdev->dev, &imx_tve_ops);  	return 0;  } diff --git a/drivers/staging/imx-drm/ipu-v3/Makefile b/drivers/staging/imx-drm/ipu-v3/Makefile deleted file mode 100644 index 28ed72e98a9..00000000000 --- a/drivers/staging/imx-drm/ipu-v3/Makefile +++ /dev/null @@ -1,3 +0,0 @@ -obj-$(CONFIG_DRM_IMX_IPUV3_CORE) += imx-ipu-v3.o - -imx-ipu-v3-objs := ipu-common.o ipu-dc.o ipu-di.o ipu-dp.o ipu-dmfc.o diff --git a/drivers/staging/imx-drm/ipu-v3/imx-ipu-v3.h b/drivers/staging/imx-drm/ipu-v3/imx-ipu-v3.h deleted file mode 100644 index 74c022e2a53..00000000000 --- a/drivers/staging/imx-drm/ipu-v3/imx-ipu-v3.h +++ /dev/null @@ -1,322 +0,0 @@ -/* - * Copyright 2005-2009 Freescale Semiconductor, Inc. - * - * The code contained herein is licensed under the GNU Lesser General - * Public License.  You may obtain a copy of the GNU Lesser General - * Public License Version 2.1 or later at the following locations: - * - * http://www.opensource.org/licenses/lgpl-license.html - * http://www.gnu.org/copyleft/lgpl.html - */ - -#ifndef __DRM_IPU_H__ -#define __DRM_IPU_H__ - -#include <linux/types.h> -#include <linux/videodev2.h> -#include <linux/bitmap.h> -#include <linux/fb.h> - -struct ipu_soc; - -enum ipuv3_type { -	IPUV3EX, -	IPUV3M, -	IPUV3H, -}; - -/* - * Bitfield of Display Interface signal polarities. - */ -struct ipu_di_signal_cfg { -	unsigned datamask_en:1; -	unsigned interlaced:1; -	unsigned odd_field_first:1; -	unsigned clksel_en:1; -	unsigned clkidle_en:1; -	unsigned data_pol:1;	/* true = inverted */ -	unsigned clk_pol:1;	/* true = rising edge */ -	unsigned enable_pol:1; -	unsigned Hsync_pol:1;	/* true = active high */ -	unsigned Vsync_pol:1; - -	u16 width; -	u16 height; -	u32 pixel_fmt; -	u16 h_start_width; -	u16 h_sync_width; -	u16 h_end_width; -	u16 v_start_width; -	u16 v_sync_width; -	u16 v_end_width; -	u32 v_to_h_sync; -	unsigned long pixelclock; -#define IPU_DI_CLKMODE_SYNC	(1 << 0) -#define IPU_DI_CLKMODE_EXT	(1 << 1) -	unsigned long clkflags; - -	u8 hsync_pin; -	u8 vsync_pin; -}; - -enum ipu_color_space { -	IPUV3_COLORSPACE_RGB, -	IPUV3_COLORSPACE_YUV, -	IPUV3_COLORSPACE_UNKNOWN, -}; - -struct ipuv3_channel; - -enum ipu_channel_irq { -	IPU_IRQ_EOF = 0, -	IPU_IRQ_NFACK = 64, -	IPU_IRQ_NFB4EOF = 128, -	IPU_IRQ_EOS = 192, -}; - -int ipu_idmac_channel_irq(struct ipu_soc *ipu, struct ipuv3_channel *channel, -		enum ipu_channel_irq irq); - -#define IPU_IRQ_DP_SF_START		(448 + 2) -#define IPU_IRQ_DP_SF_END		(448 + 3) -#define IPU_IRQ_BG_SF_END		IPU_IRQ_DP_SF_END, -#define IPU_IRQ_DC_FC_0			(448 + 8) -#define IPU_IRQ_DC_FC_1			(448 + 9) -#define IPU_IRQ_DC_FC_2			(448 + 10) -#define IPU_IRQ_DC_FC_3			(448 + 11) -#define IPU_IRQ_DC_FC_4			(448 + 12) -#define IPU_IRQ_DC_FC_6			(448 + 13) -#define IPU_IRQ_VSYNC_PRE_0		(448 + 14) -#define IPU_IRQ_VSYNC_PRE_1		(448 + 15) - -/* - * IPU Image DMA Controller (idmac) functions - */ -struct ipuv3_channel *ipu_idmac_get(struct ipu_soc *ipu, unsigned channel); -void ipu_idmac_put(struct ipuv3_channel *); - -int ipu_idmac_enable_channel(struct ipuv3_channel *channel); -int ipu_idmac_disable_channel(struct ipuv3_channel *channel); - -void ipu_idmac_set_double_buffer(struct ipuv3_channel *channel, -		bool doublebuffer); -void ipu_idmac_select_buffer(struct ipuv3_channel *channel, u32 buf_num); - -/* - * IPU Display Controller (dc) functions - */ -struct ipu_dc; -struct ipu_di; -struct ipu_dc *ipu_dc_get(struct ipu_soc *ipu, int channel); -void ipu_dc_put(struct ipu_dc *dc); -int ipu_dc_init_sync(struct ipu_dc *dc, struct ipu_di *di, bool interlaced, -		u32 pixel_fmt, u32 width); -void ipu_dc_enable_channel(struct ipu_dc *dc); -void ipu_dc_disable_channel(struct ipu_dc *dc); - -/* - * IPU Display Interface (di) functions - */ -struct ipu_di *ipu_di_get(struct ipu_soc *ipu, int disp); -void ipu_di_put(struct ipu_di *); -int ipu_di_disable(struct ipu_di *); -int ipu_di_enable(struct ipu_di *); -int ipu_di_get_num(struct ipu_di *); -int ipu_di_init_sync_panel(struct ipu_di *, struct ipu_di_signal_cfg *sig); - -/* - * IPU Display Multi FIFO Controller (dmfc) functions - */ -struct dmfc_channel; -int ipu_dmfc_enable_channel(struct dmfc_channel *dmfc); -void ipu_dmfc_disable_channel(struct dmfc_channel *dmfc); -int ipu_dmfc_alloc_bandwidth(struct dmfc_channel *dmfc, -		unsigned long bandwidth_mbs, int burstsize); -void ipu_dmfc_free_bandwidth(struct dmfc_channel *dmfc); -int ipu_dmfc_init_channel(struct dmfc_channel *dmfc, int width); -struct dmfc_channel *ipu_dmfc_get(struct ipu_soc *ipu, int ipuv3_channel); -void ipu_dmfc_put(struct dmfc_channel *dmfc); - -/* - * IPU Display Processor (dp) functions - */ -#define IPU_DP_FLOW_SYNC_BG	0 -#define IPU_DP_FLOW_SYNC_FG	1 -#define IPU_DP_FLOW_ASYNC0_BG	2 -#define IPU_DP_FLOW_ASYNC0_FG	3 -#define IPU_DP_FLOW_ASYNC1_BG	4 -#define IPU_DP_FLOW_ASYNC1_FG	5 - -struct ipu_dp *ipu_dp_get(struct ipu_soc *ipu, unsigned int flow); -void ipu_dp_put(struct ipu_dp *); -int ipu_dp_enable_channel(struct ipu_dp *dp); -void ipu_dp_disable_channel(struct ipu_dp *dp); -int ipu_dp_setup_channel(struct ipu_dp *dp, -		enum ipu_color_space in, enum ipu_color_space out); -int ipu_dp_set_window_pos(struct ipu_dp *, u16 x_pos, u16 y_pos); -int ipu_dp_set_global_alpha(struct ipu_dp *dp, bool enable, u8 alpha, -		bool bg_chan); - -#define IPU_CPMEM_WORD(word, ofs, size) ((((word) * 160 + (ofs)) << 8) | (size)) - -#define IPU_FIELD_UBO		IPU_CPMEM_WORD(0, 46, 22) -#define IPU_FIELD_VBO		IPU_CPMEM_WORD(0, 68, 22) -#define IPU_FIELD_IOX		IPU_CPMEM_WORD(0, 90, 4) -#define IPU_FIELD_RDRW		IPU_CPMEM_WORD(0, 94, 1) -#define IPU_FIELD_SO		IPU_CPMEM_WORD(0, 113, 1) -#define IPU_FIELD_SLY		IPU_CPMEM_WORD(1, 102, 14) -#define IPU_FIELD_SLUV		IPU_CPMEM_WORD(1, 128, 14) - -#define IPU_FIELD_XV		IPU_CPMEM_WORD(0, 0, 10) -#define IPU_FIELD_YV		IPU_CPMEM_WORD(0, 10, 9) -#define IPU_FIELD_XB		IPU_CPMEM_WORD(0, 19, 13) -#define IPU_FIELD_YB		IPU_CPMEM_WORD(0, 32, 12) -#define IPU_FIELD_NSB_B		IPU_CPMEM_WORD(0, 44, 1) -#define IPU_FIELD_CF		IPU_CPMEM_WORD(0, 45, 1) -#define IPU_FIELD_SX		IPU_CPMEM_WORD(0, 46, 12) -#define IPU_FIELD_SY		IPU_CPMEM_WORD(0, 58, 11) -#define IPU_FIELD_NS		IPU_CPMEM_WORD(0, 69, 10) -#define IPU_FIELD_SDX		IPU_CPMEM_WORD(0, 79, 7) -#define IPU_FIELD_SM		IPU_CPMEM_WORD(0, 86, 10) -#define IPU_FIELD_SCC		IPU_CPMEM_WORD(0, 96, 1) -#define IPU_FIELD_SCE		IPU_CPMEM_WORD(0, 97, 1) -#define IPU_FIELD_SDY		IPU_CPMEM_WORD(0, 98, 7) -#define IPU_FIELD_SDRX		IPU_CPMEM_WORD(0, 105, 1) -#define IPU_FIELD_SDRY		IPU_CPMEM_WORD(0, 106, 1) -#define IPU_FIELD_BPP		IPU_CPMEM_WORD(0, 107, 3) -#define IPU_FIELD_DEC_SEL	IPU_CPMEM_WORD(0, 110, 2) -#define IPU_FIELD_DIM		IPU_CPMEM_WORD(0, 112, 1) -#define IPU_FIELD_BNDM		IPU_CPMEM_WORD(0, 114, 3) -#define IPU_FIELD_BM		IPU_CPMEM_WORD(0, 117, 2) -#define IPU_FIELD_ROT		IPU_CPMEM_WORD(0, 119, 1) -#define IPU_FIELD_HF		IPU_CPMEM_WORD(0, 120, 1) -#define IPU_FIELD_VF		IPU_CPMEM_WORD(0, 121, 1) -#define IPU_FIELD_THE		IPU_CPMEM_WORD(0, 122, 1) -#define IPU_FIELD_CAP		IPU_CPMEM_WORD(0, 123, 1) -#define IPU_FIELD_CAE		IPU_CPMEM_WORD(0, 124, 1) -#define IPU_FIELD_FW		IPU_CPMEM_WORD(0, 125, 13) -#define IPU_FIELD_FH		IPU_CPMEM_WORD(0, 138, 12) -#define IPU_FIELD_EBA0		IPU_CPMEM_WORD(1, 0, 29) -#define IPU_FIELD_EBA1		IPU_CPMEM_WORD(1, 29, 29) -#define IPU_FIELD_ILO		IPU_CPMEM_WORD(1, 58, 20) -#define IPU_FIELD_NPB		IPU_CPMEM_WORD(1, 78, 7) -#define IPU_FIELD_PFS		IPU_CPMEM_WORD(1, 85, 4) -#define IPU_FIELD_ALU		IPU_CPMEM_WORD(1, 89, 1) -#define IPU_FIELD_ALBM		IPU_CPMEM_WORD(1, 90, 3) -#define IPU_FIELD_ID		IPU_CPMEM_WORD(1, 93, 2) -#define IPU_FIELD_TH		IPU_CPMEM_WORD(1, 95, 7) -#define IPU_FIELD_SL		IPU_CPMEM_WORD(1, 102, 14) -#define IPU_FIELD_WID0		IPU_CPMEM_WORD(1, 116, 3) -#define IPU_FIELD_WID1		IPU_CPMEM_WORD(1, 119, 3) -#define IPU_FIELD_WID2		IPU_CPMEM_WORD(1, 122, 3) -#define IPU_FIELD_WID3		IPU_CPMEM_WORD(1, 125, 3) -#define IPU_FIELD_OFS0		IPU_CPMEM_WORD(1, 128, 5) -#define IPU_FIELD_OFS1		IPU_CPMEM_WORD(1, 133, 5) -#define IPU_FIELD_OFS2		IPU_CPMEM_WORD(1, 138, 5) -#define IPU_FIELD_OFS3		IPU_CPMEM_WORD(1, 143, 5) -#define IPU_FIELD_SXYS		IPU_CPMEM_WORD(1, 148, 1) -#define IPU_FIELD_CRE		IPU_CPMEM_WORD(1, 149, 1) -#define IPU_FIELD_DEC_SEL2	IPU_CPMEM_WORD(1, 150, 1) - -struct ipu_cpmem_word { -	u32 data[5]; -	u32 res[3]; -}; - -struct ipu_ch_param { -	struct ipu_cpmem_word word[2]; -}; - -void ipu_ch_param_write_field(struct ipu_ch_param __iomem *base, u32 wbs, u32 v); -u32 ipu_ch_param_read_field(struct ipu_ch_param __iomem *base, u32 wbs); -struct ipu_ch_param __iomem *ipu_get_cpmem(struct ipuv3_channel *channel); -void ipu_ch_param_dump(struct ipu_ch_param __iomem *p); - -static inline void ipu_ch_param_zero(struct ipu_ch_param __iomem *p) -{ -	int i; -	void __iomem *base = p; - -	for (i = 0; i < sizeof(*p) / sizeof(u32); i++) -		writel(0, base + i * sizeof(u32)); -} - -static inline void ipu_cpmem_set_buffer(struct ipu_ch_param __iomem *p, -		int bufnum, dma_addr_t buf) -{ -	if (bufnum) -		ipu_ch_param_write_field(p, IPU_FIELD_EBA1, buf >> 3); -	else -		ipu_ch_param_write_field(p, IPU_FIELD_EBA0, buf >> 3); -} - -static inline void ipu_cpmem_set_resolution(struct ipu_ch_param __iomem *p, -		int xres, int yres) -{ -	ipu_ch_param_write_field(p, IPU_FIELD_FW, xres - 1); -	ipu_ch_param_write_field(p, IPU_FIELD_FH, yres - 1); -} - -static inline void ipu_cpmem_set_stride(struct ipu_ch_param __iomem *p, -		int stride) -{ -	ipu_ch_param_write_field(p, IPU_FIELD_SLY, stride - 1); -} - -void ipu_cpmem_set_high_priority(struct ipuv3_channel *channel); - -struct ipu_rgb { -	struct fb_bitfield	red; -	struct fb_bitfield	green; -	struct fb_bitfield	blue; -	struct fb_bitfield	transp; -	int			bits_per_pixel; -}; - -struct ipu_image { -	struct v4l2_pix_format pix; -	struct v4l2_rect rect; -	dma_addr_t phys; -}; - -int ipu_cpmem_set_format_passthrough(struct ipu_ch_param __iomem *p, -		int width); - -int ipu_cpmem_set_format_rgb(struct ipu_ch_param __iomem *, -		struct ipu_rgb *rgb); - -static inline void ipu_cpmem_interlaced_scan(struct ipu_ch_param *p, -		int stride) -{ -	ipu_ch_param_write_field(p, IPU_FIELD_SO, 1); -	ipu_ch_param_write_field(p, IPU_FIELD_ILO, stride / 8); -	ipu_ch_param_write_field(p, IPU_FIELD_SLY, (stride * 2) - 1); -}; - -void ipu_cpmem_set_yuv_planar(struct ipu_ch_param __iomem *p, u32 pixel_format, -			int stride, int height); -void ipu_cpmem_set_yuv_interleaved(struct ipu_ch_param __iomem *p, -				   u32 pixel_format); -void ipu_cpmem_set_yuv_planar_full(struct ipu_ch_param __iomem *p, -		u32 pixel_format, int stride, int u_offset, int v_offset); -int ipu_cpmem_set_fmt(struct ipu_ch_param __iomem *cpmem, u32 pixelformat); -int ipu_cpmem_set_image(struct ipu_ch_param __iomem *cpmem, -		struct ipu_image *image); - -enum ipu_color_space ipu_pixelformat_to_colorspace(u32 pixelformat); - -static inline void ipu_cpmem_set_burstsize(struct ipu_ch_param __iomem *p, -		int burstsize) -{ -	ipu_ch_param_write_field(p, IPU_FIELD_NPB, burstsize - 1); -}; - -struct ipu_client_platformdata { -	int di; -	int dc; -	int dp; -	int dmfc; -	int dma[2]; -}; - -#endif /* __DRM_IPU_H__ */ diff --git a/drivers/staging/imx-drm/ipu-v3/ipu-common.c b/drivers/staging/imx-drm/ipu-v3/ipu-common.c deleted file mode 100644 index ba464e5d9f1..00000000000 --- a/drivers/staging/imx-drm/ipu-v3/ipu-common.c +++ /dev/null @@ -1,1169 +0,0 @@ -/* - * Copyright (c) 2010 Sascha Hauer <s.hauer@pengutronix.de> - * Copyright (C) 2005-2009 Freescale Semiconductor, Inc. - * - * This program is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License as published by the - * Free Software Foundation; either version 2 of the License, or (at your - * option) any later version. - * - * This program is distributed in the hope that it will be useful, but - * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY - * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License - * for more details. - */ -#include <linux/module.h> -#include <linux/export.h> -#include <linux/types.h> -#include <linux/init.h> -#include <linux/reset.h> -#include <linux/platform_device.h> -#include <linux/err.h> -#include <linux/spinlock.h> -#include <linux/delay.h> -#include <linux/interrupt.h> -#include <linux/io.h> -#include <linux/clk.h> -#include <linux/list.h> -#include <linux/irq.h> -#include <linux/irqchip/chained_irq.h> -#include <linux/irqdomain.h> -#include <linux/of_device.h> - -#include "imx-ipu-v3.h" -#include "ipu-prv.h" - -static inline u32 ipu_cm_read(struct ipu_soc *ipu, unsigned offset) -{ -	return readl(ipu->cm_reg + offset); -} - -static inline void ipu_cm_write(struct ipu_soc *ipu, u32 value, unsigned offset) -{ -	writel(value, ipu->cm_reg + offset); -} - -static inline u32 ipu_idmac_read(struct ipu_soc *ipu, unsigned offset) -{ -	return readl(ipu->idmac_reg + offset); -} - -static inline void ipu_idmac_write(struct ipu_soc *ipu, u32 value, -		unsigned offset) -{ -	writel(value, ipu->idmac_reg + offset); -} - -void ipu_srm_dp_sync_update(struct ipu_soc *ipu) -{ -	u32 val; - -	val = ipu_cm_read(ipu, IPU_SRM_PRI2); -	val |= 0x8; -	ipu_cm_write(ipu, val, IPU_SRM_PRI2); -} -EXPORT_SYMBOL_GPL(ipu_srm_dp_sync_update); - -struct ipu_ch_param __iomem *ipu_get_cpmem(struct ipuv3_channel *channel) -{ -	struct ipu_soc *ipu = channel->ipu; - -	return ipu->cpmem_base + channel->num; -} -EXPORT_SYMBOL_GPL(ipu_get_cpmem); - -void ipu_cpmem_set_high_priority(struct ipuv3_channel *channel) -{ -	struct ipu_soc *ipu = channel->ipu; -	struct ipu_ch_param __iomem *p = ipu_get_cpmem(channel); -	u32 val; - -	if (ipu->ipu_type == IPUV3EX) -		ipu_ch_param_write_field(p, IPU_FIELD_ID, 1); - -	val = ipu_idmac_read(ipu, IDMAC_CHA_PRI(channel->num)); -	val |= 1 << (channel->num % 32); -	ipu_idmac_write(ipu, val, IDMAC_CHA_PRI(channel->num)); -}; -EXPORT_SYMBOL_GPL(ipu_cpmem_set_high_priority); - -void ipu_ch_param_write_field(struct ipu_ch_param __iomem *base, u32 wbs, u32 v) -{ -	u32 bit = (wbs >> 8) % 160; -	u32 size = wbs & 0xff; -	u32 word = (wbs >> 8) / 160; -	u32 i = bit / 32; -	u32 ofs = bit % 32; -	u32 mask = (1 << size) - 1; -	u32 val; - -	pr_debug("%s %d %d %d\n", __func__, word, bit , size); - -	val = readl(&base->word[word].data[i]); -	val &= ~(mask << ofs); -	val |= v << ofs; -	writel(val, &base->word[word].data[i]); - -	if ((bit + size - 1) / 32 > i) { -		val = readl(&base->word[word].data[i + 1]); -		val &= ~(mask >> (ofs ? (32 - ofs) : 0)); -		val |= v >> (ofs ? (32 - ofs) : 0); -		writel(val, &base->word[word].data[i + 1]); -	} -} -EXPORT_SYMBOL_GPL(ipu_ch_param_write_field); - -u32 ipu_ch_param_read_field(struct ipu_ch_param __iomem *base, u32 wbs) -{ -	u32 bit = (wbs >> 8) % 160; -	u32 size = wbs & 0xff; -	u32 word = (wbs >> 8) / 160; -	u32 i = bit / 32; -	u32 ofs = bit % 32; -	u32 mask = (1 << size) - 1; -	u32 val = 0; - -	pr_debug("%s %d %d %d\n", __func__, word, bit , size); - -	val = (readl(&base->word[word].data[i]) >> ofs) & mask; - -	if ((bit + size - 1) / 32 > i) { -		u32 tmp; -		tmp = readl(&base->word[word].data[i + 1]); -		tmp &= mask >> (ofs ? (32 - ofs) : 0); -		val |= tmp << (ofs ? (32 - ofs) : 0); -	} - -	return val; -} -EXPORT_SYMBOL_GPL(ipu_ch_param_read_field); - -int ipu_cpmem_set_format_rgb(struct ipu_ch_param __iomem *p, -		struct ipu_rgb *rgb) -{ -	int bpp = 0, npb = 0, ro, go, bo, to; - -	ro = rgb->bits_per_pixel - rgb->red.length - rgb->red.offset; -	go = rgb->bits_per_pixel - rgb->green.length - rgb->green.offset; -	bo = rgb->bits_per_pixel - rgb->blue.length - rgb->blue.offset; -	to = rgb->bits_per_pixel - rgb->transp.length - rgb->transp.offset; - -	ipu_ch_param_write_field(p, IPU_FIELD_WID0, rgb->red.length - 1); -	ipu_ch_param_write_field(p, IPU_FIELD_OFS0, ro); -	ipu_ch_param_write_field(p, IPU_FIELD_WID1, rgb->green.length - 1); -	ipu_ch_param_write_field(p, IPU_FIELD_OFS1, go); -	ipu_ch_param_write_field(p, IPU_FIELD_WID2, rgb->blue.length - 1); -	ipu_ch_param_write_field(p, IPU_FIELD_OFS2, bo); - -	if (rgb->transp.length) { -		ipu_ch_param_write_field(p, IPU_FIELD_WID3, -				rgb->transp.length - 1); -		ipu_ch_param_write_field(p, IPU_FIELD_OFS3, to); -	} else { -		ipu_ch_param_write_field(p, IPU_FIELD_WID3, 7); -		ipu_ch_param_write_field(p, IPU_FIELD_OFS3, -				rgb->bits_per_pixel); -	} - -	switch (rgb->bits_per_pixel) { -	case 32: -		bpp = 0; -		npb = 15; -		break; -	case 24: -		bpp = 1; -		npb = 19; -		break; -	case 16: -		bpp = 3; -		npb = 31; -		break; -	case 8: -		bpp = 5; -		npb = 63; -		break; -	default: -		return -EINVAL; -	} -	ipu_ch_param_write_field(p, IPU_FIELD_BPP, bpp); -	ipu_ch_param_write_field(p, IPU_FIELD_NPB, npb); -	ipu_ch_param_write_field(p, IPU_FIELD_PFS, 7); /* rgb mode */ - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_cpmem_set_format_rgb); - -int ipu_cpmem_set_format_passthrough(struct ipu_ch_param __iomem *p, -		int width) -{ -	int bpp = 0, npb = 0; - -	switch (width) { -	case 32: -		bpp = 0; -		npb = 15; -		break; -	case 24: -		bpp = 1; -		npb = 19; -		break; -	case 16: -		bpp = 3; -		npb = 31; -		break; -	case 8: -		bpp = 5; -		npb = 63; -		break; -	default: -		return -EINVAL; -	} - -	ipu_ch_param_write_field(p, IPU_FIELD_BPP, bpp); -	ipu_ch_param_write_field(p, IPU_FIELD_NPB, npb); -	ipu_ch_param_write_field(p, IPU_FIELD_PFS, 6); /* raw mode */ - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_cpmem_set_format_passthrough); - -void ipu_cpmem_set_yuv_interleaved(struct ipu_ch_param __iomem *p, -				   u32 pixel_format) -{ -	switch (pixel_format) { -	case V4L2_PIX_FMT_UYVY: -		ipu_ch_param_write_field(p, IPU_FIELD_BPP, 3);    /* bits/pixel */ -		ipu_ch_param_write_field(p, IPU_FIELD_PFS, 0xA);  /* pix format */ -		ipu_ch_param_write_field(p, IPU_FIELD_NPB, 31);   /* burst size */ -		break; -	case V4L2_PIX_FMT_YUYV: -		ipu_ch_param_write_field(p, IPU_FIELD_BPP, 3);    /* bits/pixel */ -		ipu_ch_param_write_field(p, IPU_FIELD_PFS, 0x8);  /* pix format */ -		ipu_ch_param_write_field(p, IPU_FIELD_NPB, 31);   /* burst size */ -		break; -	} -} -EXPORT_SYMBOL_GPL(ipu_cpmem_set_yuv_interleaved); - -void ipu_cpmem_set_yuv_planar_full(struct ipu_ch_param __iomem *p, -		u32 pixel_format, int stride, int u_offset, int v_offset) -{ -	switch (pixel_format) { -	case V4L2_PIX_FMT_YUV420: -		ipu_ch_param_write_field(p, IPU_FIELD_SLUV, (stride / 2) - 1); -		ipu_ch_param_write_field(p, IPU_FIELD_UBO, u_offset / 8); -		ipu_ch_param_write_field(p, IPU_FIELD_VBO, v_offset / 8); -		break; -	case V4L2_PIX_FMT_YVU420: -		ipu_ch_param_write_field(p, IPU_FIELD_SLUV, (stride / 2) - 1); -		ipu_ch_param_write_field(p, IPU_FIELD_UBO, v_offset / 8); -		ipu_ch_param_write_field(p, IPU_FIELD_VBO, u_offset / 8); -		break; -	} -} -EXPORT_SYMBOL_GPL(ipu_cpmem_set_yuv_planar_full); - -void ipu_cpmem_set_yuv_planar(struct ipu_ch_param __iomem *p, u32 pixel_format, -		int stride, int height) -{ -	int u_offset, v_offset; -	int uv_stride = 0; - -	switch (pixel_format) { -	case V4L2_PIX_FMT_YUV420: -	case V4L2_PIX_FMT_YVU420: -		uv_stride = stride / 2; -		u_offset = stride * height; -		v_offset = u_offset + (uv_stride * height / 2); -		ipu_cpmem_set_yuv_planar_full(p, pixel_format, stride, -				u_offset, v_offset); -		break; -	} -} -EXPORT_SYMBOL_GPL(ipu_cpmem_set_yuv_planar); - -static struct ipu_rgb def_rgb_32 = { -	.red	= { .offset = 16, .length = 8, }, -	.green	= { .offset =  8, .length = 8, }, -	.blue	= { .offset =  0, .length = 8, }, -	.transp = { .offset = 24, .length = 8, }, -	.bits_per_pixel = 32, -}; - -static struct ipu_rgb def_bgr_32 = { -	.red	= { .offset = 16, .length = 8, }, -	.green	= { .offset =  8, .length = 8, }, -	.blue	= { .offset =  0, .length = 8, }, -	.transp = { .offset = 24, .length = 8, }, -	.bits_per_pixel = 32, -}; - -static struct ipu_rgb def_rgb_24 = { -	.red	= { .offset =  0, .length = 8, }, -	.green	= { .offset =  8, .length = 8, }, -	.blue	= { .offset = 16, .length = 8, }, -	.transp = { .offset =  0, .length = 0, }, -	.bits_per_pixel = 24, -}; - -static struct ipu_rgb def_bgr_24 = { -	.red	= { .offset = 16, .length = 8, }, -	.green	= { .offset =  8, .length = 8, }, -	.blue	= { .offset =  0, .length = 8, }, -	.transp = { .offset =  0, .length = 0, }, -	.bits_per_pixel = 24, -}; - -static struct ipu_rgb def_rgb_16 = { -	.red	= { .offset = 11, .length = 5, }, -	.green	= { .offset =  5, .length = 6, }, -	.blue	= { .offset =  0, .length = 5, }, -	.transp = { .offset =  0, .length = 0, }, -	.bits_per_pixel = 16, -}; - -#define Y_OFFSET(pix, x, y)	((x) + pix->width * (y)) -#define U_OFFSET(pix, x, y)	((pix->width * pix->height) + \ -					(pix->width * (y) / 4) + (x) / 2) -#define V_OFFSET(pix, x, y)	((pix->width * pix->height) + \ -					(pix->width * pix->height / 4) + \ -					(pix->width * (y) / 4) + (x) / 2) - -int ipu_cpmem_set_fmt(struct ipu_ch_param __iomem *cpmem, u32 pixelformat) -{ -	switch (pixelformat) { -	case V4L2_PIX_FMT_YUV420: -	case V4L2_PIX_FMT_YVU420: -		/* pix format */ -		ipu_ch_param_write_field(cpmem, IPU_FIELD_PFS, 2); -		/* burst size */ -		ipu_ch_param_write_field(cpmem, IPU_FIELD_NPB, 63); -		break; -	case V4L2_PIX_FMT_UYVY: -		/* bits/pixel */ -		ipu_ch_param_write_field(cpmem, IPU_FIELD_BPP, 3); -		/* pix format */ -		ipu_ch_param_write_field(cpmem, IPU_FIELD_PFS, 0xA); -		/* burst size */ -		ipu_ch_param_write_field(cpmem, IPU_FIELD_NPB, 31); -		break; -	case V4L2_PIX_FMT_YUYV: -		/* bits/pixel */ -		ipu_ch_param_write_field(cpmem, IPU_FIELD_BPP, 3); -		/* pix format */ -		ipu_ch_param_write_field(cpmem, IPU_FIELD_PFS, 0x8); -		/* burst size */ -		ipu_ch_param_write_field(cpmem, IPU_FIELD_NPB, 31); -		break; -	case V4L2_PIX_FMT_RGB32: -		ipu_cpmem_set_format_rgb(cpmem, &def_rgb_32); -		break; -	case V4L2_PIX_FMT_RGB565: -		ipu_cpmem_set_format_rgb(cpmem, &def_rgb_16); -		break; -	case V4L2_PIX_FMT_BGR32: -		ipu_cpmem_set_format_rgb(cpmem, &def_bgr_32); -		break; -	case V4L2_PIX_FMT_RGB24: -		ipu_cpmem_set_format_rgb(cpmem, &def_rgb_24); -		break; -	case V4L2_PIX_FMT_BGR24: -		ipu_cpmem_set_format_rgb(cpmem, &def_bgr_24); -		break; -	default: -		return -EINVAL; -	} - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_cpmem_set_fmt); - -int ipu_cpmem_set_image(struct ipu_ch_param __iomem *cpmem, -		struct ipu_image *image) -{ -	struct v4l2_pix_format *pix = &image->pix; -	int y_offset, u_offset, v_offset; - -	pr_debug("%s: resolution: %dx%d stride: %d\n", -			__func__, pix->width, pix->height, -			pix->bytesperline); - -	ipu_cpmem_set_resolution(cpmem, image->rect.width, -			image->rect.height); -	ipu_cpmem_set_stride(cpmem, pix->bytesperline); - -	ipu_cpmem_set_fmt(cpmem, pix->pixelformat); - -	switch (pix->pixelformat) { -	case V4L2_PIX_FMT_YUV420: -	case V4L2_PIX_FMT_YVU420: -		y_offset = Y_OFFSET(pix, image->rect.left, image->rect.top); -		u_offset = U_OFFSET(pix, image->rect.left, -				image->rect.top) - y_offset; -		v_offset = V_OFFSET(pix, image->rect.left, -				image->rect.top) - y_offset; - -		ipu_cpmem_set_yuv_planar_full(cpmem, pix->pixelformat, -				pix->bytesperline, u_offset, v_offset); -		ipu_cpmem_set_buffer(cpmem, 0, image->phys + y_offset); -		break; -	case V4L2_PIX_FMT_UYVY: -	case V4L2_PIX_FMT_YUYV: -		ipu_cpmem_set_buffer(cpmem, 0, image->phys + -				image->rect.left * 2 + -				image->rect.top * image->pix.bytesperline); -		break; -	case V4L2_PIX_FMT_RGB32: -	case V4L2_PIX_FMT_BGR32: -		ipu_cpmem_set_buffer(cpmem, 0, image->phys + -				image->rect.left * 4 + -				image->rect.top * image->pix.bytesperline); -		break; -	case V4L2_PIX_FMT_RGB565: -		ipu_cpmem_set_buffer(cpmem, 0, image->phys + -				image->rect.left * 2 + -				image->rect.top * image->pix.bytesperline); -		break; -	case V4L2_PIX_FMT_RGB24: -	case V4L2_PIX_FMT_BGR24: -		ipu_cpmem_set_buffer(cpmem, 0, image->phys + -				image->rect.left * 3 + -				image->rect.top * image->pix.bytesperline); -		break; -	default: -		return -EINVAL; -	} - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_cpmem_set_image); - -enum ipu_color_space ipu_pixelformat_to_colorspace(u32 pixelformat) -{ -	switch (pixelformat) { -	case V4L2_PIX_FMT_YUV420: -	case V4L2_PIX_FMT_YVU420: -	case V4L2_PIX_FMT_UYVY: -	case V4L2_PIX_FMT_YUYV: -		return IPUV3_COLORSPACE_YUV; -	case V4L2_PIX_FMT_RGB32: -	case V4L2_PIX_FMT_BGR32: -	case V4L2_PIX_FMT_RGB24: -	case V4L2_PIX_FMT_BGR24: -	case V4L2_PIX_FMT_RGB565: -		return IPUV3_COLORSPACE_RGB; -	default: -		return IPUV3_COLORSPACE_UNKNOWN; -	} -} -EXPORT_SYMBOL_GPL(ipu_pixelformat_to_colorspace); - -struct ipuv3_channel *ipu_idmac_get(struct ipu_soc *ipu, unsigned num) -{ -	struct ipuv3_channel *channel; - -	dev_dbg(ipu->dev, "%s %d\n", __func__, num); - -	if (num > 63) -		return ERR_PTR(-ENODEV); - -	mutex_lock(&ipu->channel_lock); - -	channel = &ipu->channel[num]; - -	if (channel->busy) { -		channel = ERR_PTR(-EBUSY); -		goto out; -	} - -	channel->busy = 1; -	channel->num = num; - -out: -	mutex_unlock(&ipu->channel_lock); - -	return channel; -} -EXPORT_SYMBOL_GPL(ipu_idmac_get); - -void ipu_idmac_put(struct ipuv3_channel *channel) -{ -	struct ipu_soc *ipu = channel->ipu; - -	dev_dbg(ipu->dev, "%s %d\n", __func__, channel->num); - -	mutex_lock(&ipu->channel_lock); - -	channel->busy = 0; - -	mutex_unlock(&ipu->channel_lock); -} -EXPORT_SYMBOL_GPL(ipu_idmac_put); - -#define idma_mask(ch)			(1 << (ch & 0x1f)) - -void ipu_idmac_set_double_buffer(struct ipuv3_channel *channel, -		bool doublebuffer) -{ -	struct ipu_soc *ipu = channel->ipu; -	unsigned long flags; -	u32 reg; - -	spin_lock_irqsave(&ipu->lock, flags); - -	reg = ipu_cm_read(ipu, IPU_CHA_DB_MODE_SEL(channel->num)); -	if (doublebuffer) -		reg |= idma_mask(channel->num); -	else -		reg &= ~idma_mask(channel->num); -	ipu_cm_write(ipu, reg, IPU_CHA_DB_MODE_SEL(channel->num)); - -	spin_unlock_irqrestore(&ipu->lock, flags); -} -EXPORT_SYMBOL_GPL(ipu_idmac_set_double_buffer); - -int ipu_module_enable(struct ipu_soc *ipu, u32 mask) -{ -	unsigned long lock_flags; -	u32 val; - -	spin_lock_irqsave(&ipu->lock, lock_flags); - -	val = ipu_cm_read(ipu, IPU_DISP_GEN); - -	if (mask & IPU_CONF_DI0_EN) -		val |= IPU_DI0_COUNTER_RELEASE; -	if (mask & IPU_CONF_DI1_EN) -		val |= IPU_DI1_COUNTER_RELEASE; - -	ipu_cm_write(ipu, val, IPU_DISP_GEN); - -	val = ipu_cm_read(ipu, IPU_CONF); -	val |= mask; -	ipu_cm_write(ipu, val, IPU_CONF); - -	spin_unlock_irqrestore(&ipu->lock, lock_flags); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_module_enable); - -int ipu_module_disable(struct ipu_soc *ipu, u32 mask) -{ -	unsigned long lock_flags; -	u32 val; - -	spin_lock_irqsave(&ipu->lock, lock_flags); - -	val = ipu_cm_read(ipu, IPU_CONF); -	val &= ~mask; -	ipu_cm_write(ipu, val, IPU_CONF); - -	val = ipu_cm_read(ipu, IPU_DISP_GEN); - -	if (mask & IPU_CONF_DI0_EN) -		val &= ~IPU_DI0_COUNTER_RELEASE; -	if (mask & IPU_CONF_DI1_EN) -		val &= ~IPU_DI1_COUNTER_RELEASE; - -	ipu_cm_write(ipu, val, IPU_DISP_GEN); - -	spin_unlock_irqrestore(&ipu->lock, lock_flags); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_module_disable); - -void ipu_idmac_select_buffer(struct ipuv3_channel *channel, u32 buf_num) -{ -	struct ipu_soc *ipu = channel->ipu; -	unsigned int chno = channel->num; -	unsigned long flags; - -	spin_lock_irqsave(&ipu->lock, flags); - -	/* Mark buffer as ready. */ -	if (buf_num == 0) -		ipu_cm_write(ipu, idma_mask(chno), IPU_CHA_BUF0_RDY(chno)); -	else -		ipu_cm_write(ipu, idma_mask(chno), IPU_CHA_BUF1_RDY(chno)); - -	spin_unlock_irqrestore(&ipu->lock, flags); -} -EXPORT_SYMBOL_GPL(ipu_idmac_select_buffer); - -int ipu_idmac_enable_channel(struct ipuv3_channel *channel) -{ -	struct ipu_soc *ipu = channel->ipu; -	u32 val; -	unsigned long flags; - -	spin_lock_irqsave(&ipu->lock, flags); - -	val = ipu_idmac_read(ipu, IDMAC_CHA_EN(channel->num)); -	val |= idma_mask(channel->num); -	ipu_idmac_write(ipu, val, IDMAC_CHA_EN(channel->num)); - -	spin_unlock_irqrestore(&ipu->lock, flags); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_idmac_enable_channel); - -int ipu_idmac_disable_channel(struct ipuv3_channel *channel) -{ -	struct ipu_soc *ipu = channel->ipu; -	u32 val; -	unsigned long flags; -	unsigned long timeout; - -	timeout = jiffies + msecs_to_jiffies(50); -	while (ipu_idmac_read(ipu, IDMAC_CHA_BUSY(channel->num)) & -			idma_mask(channel->num)) { -		if (time_after(jiffies, timeout)) { -			dev_warn(ipu->dev, "disabling busy idmac channel %d\n", -					channel->num); -			break; -		} -		cpu_relax(); -	} - -	spin_lock_irqsave(&ipu->lock, flags); - -	/* Disable DMA channel(s) */ -	val = ipu_idmac_read(ipu, IDMAC_CHA_EN(channel->num)); -	val &= ~idma_mask(channel->num); -	ipu_idmac_write(ipu, val, IDMAC_CHA_EN(channel->num)); - -	/* Set channel buffers NOT to be ready */ -	ipu_cm_write(ipu, 0xf0000000, IPU_GPR); /* write one to clear */ - -	if (ipu_cm_read(ipu, IPU_CHA_BUF0_RDY(channel->num)) & -			idma_mask(channel->num)) { -		ipu_cm_write(ipu, idma_mask(channel->num), -			     IPU_CHA_BUF0_RDY(channel->num)); -	} - -	if (ipu_cm_read(ipu, IPU_CHA_BUF1_RDY(channel->num)) & -			idma_mask(channel->num)) { -		ipu_cm_write(ipu, idma_mask(channel->num), -			     IPU_CHA_BUF1_RDY(channel->num)); -	} - -	ipu_cm_write(ipu, 0x0, IPU_GPR); /* write one to set */ - -	/* Reset the double buffer */ -	val = ipu_cm_read(ipu, IPU_CHA_DB_MODE_SEL(channel->num)); -	val &= ~idma_mask(channel->num); -	ipu_cm_write(ipu, val, IPU_CHA_DB_MODE_SEL(channel->num)); - -	spin_unlock_irqrestore(&ipu->lock, flags); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_idmac_disable_channel); - -static int ipu_memory_reset(struct ipu_soc *ipu) -{ -	unsigned long timeout; - -	ipu_cm_write(ipu, 0x807FFFFF, IPU_MEM_RST); - -	timeout = jiffies + msecs_to_jiffies(1000); -	while (ipu_cm_read(ipu, IPU_MEM_RST) & 0x80000000) { -		if (time_after(jiffies, timeout)) -			return -ETIME; -		cpu_relax(); -	} - -	return 0; -} - -struct ipu_devtype { -	const char *name; -	unsigned long cm_ofs; -	unsigned long cpmem_ofs; -	unsigned long srm_ofs; -	unsigned long tpm_ofs; -	unsigned long disp0_ofs; -	unsigned long disp1_ofs; -	unsigned long dc_tmpl_ofs; -	unsigned long vdi_ofs; -	enum ipuv3_type type; -}; - -static struct ipu_devtype ipu_type_imx51 = { -	.name = "IPUv3EX", -	.cm_ofs = 0x1e000000, -	.cpmem_ofs = 0x1f000000, -	.srm_ofs = 0x1f040000, -	.tpm_ofs = 0x1f060000, -	.disp0_ofs = 0x1e040000, -	.disp1_ofs = 0x1e048000, -	.dc_tmpl_ofs = 0x1f080000, -	.vdi_ofs = 0x1e068000, -	.type = IPUV3EX, -}; - -static struct ipu_devtype ipu_type_imx53 = { -	.name = "IPUv3M", -	.cm_ofs = 0x06000000, -	.cpmem_ofs = 0x07000000, -	.srm_ofs = 0x07040000, -	.tpm_ofs = 0x07060000, -	.disp0_ofs = 0x06040000, -	.disp1_ofs = 0x06048000, -	.dc_tmpl_ofs = 0x07080000, -	.vdi_ofs = 0x06068000, -	.type = IPUV3M, -}; - -static struct ipu_devtype ipu_type_imx6q = { -	.name = "IPUv3H", -	.cm_ofs = 0x00200000, -	.cpmem_ofs = 0x00300000, -	.srm_ofs = 0x00340000, -	.tpm_ofs = 0x00360000, -	.disp0_ofs = 0x00240000, -	.disp1_ofs = 0x00248000, -	.dc_tmpl_ofs = 0x00380000, -	.vdi_ofs = 0x00268000, -	.type = IPUV3H, -}; - -static const struct of_device_id imx_ipu_dt_ids[] = { -	{ .compatible = "fsl,imx51-ipu", .data = &ipu_type_imx51, }, -	{ .compatible = "fsl,imx53-ipu", .data = &ipu_type_imx53, }, -	{ .compatible = "fsl,imx6q-ipu", .data = &ipu_type_imx6q, }, -	{ /* sentinel */ } -}; -MODULE_DEVICE_TABLE(of, imx_ipu_dt_ids); - -static int ipu_submodules_init(struct ipu_soc *ipu, -		struct platform_device *pdev, unsigned long ipu_base, -		struct clk *ipu_clk) -{ -	char *unit; -	int ret; -	struct device *dev = &pdev->dev; -	const struct ipu_devtype *devtype = ipu->devtype; - -	ret = ipu_di_init(ipu, dev, 0, ipu_base + devtype->disp0_ofs, -			IPU_CONF_DI0_EN, ipu_clk); -	if (ret) { -		unit = "di0"; -		goto err_di_0; -	} - -	ret = ipu_di_init(ipu, dev, 1, ipu_base + devtype->disp1_ofs, -			IPU_CONF_DI1_EN, ipu_clk); -	if (ret) { -		unit = "di1"; -		goto err_di_1; -	} - -	ret = ipu_dc_init(ipu, dev, ipu_base + devtype->cm_ofs + -			IPU_CM_DC_REG_OFS, ipu_base + devtype->dc_tmpl_ofs); -	if (ret) { -		unit = "dc_template"; -		goto err_dc; -	} - -	ret = ipu_dmfc_init(ipu, dev, ipu_base + -			devtype->cm_ofs + IPU_CM_DMFC_REG_OFS, ipu_clk); -	if (ret) { -		unit = "dmfc"; -		goto err_dmfc; -	} - -	ret = ipu_dp_init(ipu, dev, ipu_base + devtype->srm_ofs); -	if (ret) { -		unit = "dp"; -		goto err_dp; -	} - -	return 0; - -err_dp: -	ipu_dmfc_exit(ipu); -err_dmfc: -	ipu_dc_exit(ipu); -err_dc: -	ipu_di_exit(ipu, 1); -err_di_1: -	ipu_di_exit(ipu, 0); -err_di_0: -	dev_err(&pdev->dev, "init %s failed with %d\n", unit, ret); -	return ret; -} - -static void ipu_irq_handle(struct ipu_soc *ipu, const int *regs, int num_regs) -{ -	unsigned long status; -	int i, bit, irq; - -	for (i = 0; i < num_regs; i++) { - -		status = ipu_cm_read(ipu, IPU_INT_STAT(regs[i])); -		status &= ipu_cm_read(ipu, IPU_INT_CTRL(regs[i])); - -		for_each_set_bit(bit, &status, 32) { -			irq = irq_linear_revmap(ipu->domain, regs[i] * 32 + bit); -			if (irq) -				generic_handle_irq(irq); -		} -	} -} - -static void ipu_irq_handler(unsigned int irq, struct irq_desc *desc) -{ -	struct ipu_soc *ipu = irq_desc_get_handler_data(desc); -	const int int_reg[] = { 0, 1, 2, 3, 10, 11, 12, 13, 14}; -	struct irq_chip *chip = irq_get_chip(irq); - -	chained_irq_enter(chip, desc); - -	ipu_irq_handle(ipu, int_reg, ARRAY_SIZE(int_reg)); - -	chained_irq_exit(chip, desc); -} - -static void ipu_err_irq_handler(unsigned int irq, struct irq_desc *desc) -{ -	struct ipu_soc *ipu = irq_desc_get_handler_data(desc); -	const int int_reg[] = { 4, 5, 8, 9}; -	struct irq_chip *chip = irq_get_chip(irq); - -	chained_irq_enter(chip, desc); - -	ipu_irq_handle(ipu, int_reg, ARRAY_SIZE(int_reg)); - -	chained_irq_exit(chip, desc); -} - -int ipu_idmac_channel_irq(struct ipu_soc *ipu, struct ipuv3_channel *channel, -		enum ipu_channel_irq irq_type) -{ -	int irq = irq_linear_revmap(ipu->domain, irq_type + channel->num); - -	if (!irq) -		irq = irq_create_mapping(ipu->domain, irq_type + channel->num); - -	return irq; -} -EXPORT_SYMBOL_GPL(ipu_idmac_channel_irq); - -static void ipu_submodules_exit(struct ipu_soc *ipu) -{ -	ipu_dp_exit(ipu); -	ipu_dmfc_exit(ipu); -	ipu_dc_exit(ipu); -	ipu_di_exit(ipu, 1); -	ipu_di_exit(ipu, 0); -} - -static int platform_remove_devices_fn(struct device *dev, void *unused) -{ -	struct platform_device *pdev = to_platform_device(dev); - -	platform_device_unregister(pdev); - -	return 0; -} - -static void platform_device_unregister_children(struct platform_device *pdev) -{ -	device_for_each_child(&pdev->dev, NULL, platform_remove_devices_fn); -} - -struct ipu_platform_reg { -	struct ipu_client_platformdata pdata; -	const char *name; -}; - -static const struct ipu_platform_reg client_reg[] = { -	{ -		.pdata = { -			.di = 0, -			.dc = 5, -			.dp = IPU_DP_FLOW_SYNC_BG, -			.dma[0] = IPUV3_CHANNEL_MEM_BG_SYNC, -			.dma[1] = -EINVAL, -		}, -		.name = "imx-ipuv3-crtc", -	}, { -		.pdata = { -			.di = 1, -			.dc = 1, -			.dp = -EINVAL, -			.dma[0] = IPUV3_CHANNEL_MEM_DC_SYNC, -			.dma[1] = -EINVAL, -		}, -		.name = "imx-ipuv3-crtc", -	}, -}; - -static int ipu_client_id; - -static int ipu_add_subdevice_pdata(struct device *dev, -		const struct ipu_platform_reg *reg) -{ -	struct platform_device *pdev; - -	pdev = platform_device_register_data(dev, reg->name, ipu_client_id++, -			®->pdata, sizeof(struct ipu_platform_reg)); - -	return pdev ? 0 : -EINVAL; -} - -static int ipu_add_client_devices(struct ipu_soc *ipu) -{ -	int ret; -	int i; - -	for (i = 0; i < ARRAY_SIZE(client_reg); i++) { -		const struct ipu_platform_reg *reg = &client_reg[i]; -		ret = ipu_add_subdevice_pdata(ipu->dev, reg); -		if (ret) -			goto err_register; -	} - -	return 0; - -err_register: -	platform_device_unregister_children(to_platform_device(ipu->dev)); - -	return ret; -} - - -static int ipu_irq_init(struct ipu_soc *ipu) -{ -	struct irq_chip_generic *gc; -	struct irq_chip_type *ct; -	unsigned long unused[IPU_NUM_IRQS / 32] = { -		0x400100d0, 0xffe000fd, -		0x400100d0, 0xffe000fd, -		0x400100d0, 0xffe000fd, -		0x4077ffff, 0xffe7e1fd, -		0x23fffffe, 0x8880fff0, -		0xf98fe7d0, 0xfff81fff, -		0x400100d0, 0xffe000fd, -		0x00000000, -	}; -	int ret, i; - -	ipu->domain = irq_domain_add_linear(ipu->dev->of_node, IPU_NUM_IRQS, -					    &irq_generic_chip_ops, ipu); -	if (!ipu->domain) { -		dev_err(ipu->dev, "failed to add irq domain\n"); -		return -ENODEV; -	} - -	ret = irq_alloc_domain_generic_chips(ipu->domain, 32, 1, "IPU", -					     handle_level_irq, 0, IRQF_VALID, 0); -	if (ret < 0) { -		dev_err(ipu->dev, "failed to alloc generic irq chips\n"); -		irq_domain_remove(ipu->domain); -		return ret; -	} - -	for (i = 0; i < IPU_NUM_IRQS; i += 32) { -		gc = irq_get_domain_generic_chip(ipu->domain, i); -		gc->reg_base = ipu->cm_reg; -		gc->unused = unused[i / 32]; -		ct = gc->chip_types; -		ct->chip.irq_ack = irq_gc_ack_set_bit; -		ct->chip.irq_mask = irq_gc_mask_clr_bit; -		ct->chip.irq_unmask = irq_gc_mask_set_bit; -		ct->regs.ack = IPU_INT_STAT(i / 32); -		ct->regs.mask = IPU_INT_CTRL(i / 32); -	} - -	irq_set_chained_handler(ipu->irq_sync, ipu_irq_handler); -	irq_set_handler_data(ipu->irq_sync, ipu); -	irq_set_chained_handler(ipu->irq_err, ipu_err_irq_handler); -	irq_set_handler_data(ipu->irq_err, ipu); - -	return 0; -} - -static void ipu_irq_exit(struct ipu_soc *ipu) -{ -	int i, irq; - -	irq_set_chained_handler(ipu->irq_err, NULL); -	irq_set_handler_data(ipu->irq_err, NULL); -	irq_set_chained_handler(ipu->irq_sync, NULL); -	irq_set_handler_data(ipu->irq_sync, NULL); - -	/* TODO: remove irq_domain_generic_chips */ - -	for (i = 0; i < IPU_NUM_IRQS; i++) { -		irq = irq_linear_revmap(ipu->domain, i); -		if (irq) -			irq_dispose_mapping(irq); -	} - -	irq_domain_remove(ipu->domain); -} - -static int ipu_probe(struct platform_device *pdev) -{ -	const struct of_device_id *of_id = -			of_match_device(imx_ipu_dt_ids, &pdev->dev); -	struct ipu_soc *ipu; -	struct resource *res; -	unsigned long ipu_base; -	int i, ret, irq_sync, irq_err; -	const struct ipu_devtype *devtype; - -	devtype = of_id->data; - -	irq_sync = platform_get_irq(pdev, 0); -	irq_err = platform_get_irq(pdev, 1); -	res = platform_get_resource(pdev, IORESOURCE_MEM, 0); - -	dev_dbg(&pdev->dev, "irq_sync: %d irq_err: %d\n", -			irq_sync, irq_err); - -	if (!res || irq_sync < 0 || irq_err < 0) -		return -ENODEV; - -	ipu_base = res->start; - -	ipu = devm_kzalloc(&pdev->dev, sizeof(*ipu), GFP_KERNEL); -	if (!ipu) -		return -ENODEV; - -	for (i = 0; i < 64; i++) -		ipu->channel[i].ipu = ipu; -	ipu->devtype = devtype; -	ipu->ipu_type = devtype->type; - -	spin_lock_init(&ipu->lock); -	mutex_init(&ipu->channel_lock); - -	dev_dbg(&pdev->dev, "cm_reg:   0x%08lx\n", -			ipu_base + devtype->cm_ofs); -	dev_dbg(&pdev->dev, "idmac:    0x%08lx\n", -			ipu_base + devtype->cm_ofs + IPU_CM_IDMAC_REG_OFS); -	dev_dbg(&pdev->dev, "cpmem:    0x%08lx\n", -			ipu_base + devtype->cpmem_ofs); -	dev_dbg(&pdev->dev, "disp0:    0x%08lx\n", -			ipu_base + devtype->disp0_ofs); -	dev_dbg(&pdev->dev, "disp1:    0x%08lx\n", -			ipu_base + devtype->disp1_ofs); -	dev_dbg(&pdev->dev, "srm:      0x%08lx\n", -			ipu_base + devtype->srm_ofs); -	dev_dbg(&pdev->dev, "tpm:      0x%08lx\n", -			ipu_base + devtype->tpm_ofs); -	dev_dbg(&pdev->dev, "dc:       0x%08lx\n", -			ipu_base + devtype->cm_ofs + IPU_CM_DC_REG_OFS); -	dev_dbg(&pdev->dev, "ic:       0x%08lx\n", -			ipu_base + devtype->cm_ofs + IPU_CM_IC_REG_OFS); -	dev_dbg(&pdev->dev, "dmfc:     0x%08lx\n", -			ipu_base + devtype->cm_ofs + IPU_CM_DMFC_REG_OFS); -	dev_dbg(&pdev->dev, "vdi:      0x%08lx\n", -			ipu_base + devtype->vdi_ofs); - -	ipu->cm_reg = devm_ioremap(&pdev->dev, -			ipu_base + devtype->cm_ofs, PAGE_SIZE); -	ipu->idmac_reg = devm_ioremap(&pdev->dev, -			ipu_base + devtype->cm_ofs + IPU_CM_IDMAC_REG_OFS, -			PAGE_SIZE); -	ipu->cpmem_base = devm_ioremap(&pdev->dev, -			ipu_base + devtype->cpmem_ofs, PAGE_SIZE); - -	if (!ipu->cm_reg || !ipu->idmac_reg || !ipu->cpmem_base) -		return -ENOMEM; - -	ipu->clk = devm_clk_get(&pdev->dev, "bus"); -	if (IS_ERR(ipu->clk)) { -		ret = PTR_ERR(ipu->clk); -		dev_err(&pdev->dev, "clk_get failed with %d", ret); -		return ret; -	} - -	platform_set_drvdata(pdev, ipu); - -	ret = clk_prepare_enable(ipu->clk); -	if (ret) { -		dev_err(&pdev->dev, "clk_prepare_enable failed: %d\n", ret); -		return ret; -	} - -	ipu->dev = &pdev->dev; -	ipu->irq_sync = irq_sync; -	ipu->irq_err = irq_err; - -	ret = ipu_irq_init(ipu); -	if (ret) -		goto out_failed_irq; - -	ret = device_reset(&pdev->dev); -	if (ret) { -		dev_err(&pdev->dev, "failed to reset: %d\n", ret); -		goto out_failed_reset; -	} -	ret = ipu_memory_reset(ipu); -	if (ret) -		goto out_failed_reset; - -	/* Set MCU_T to divide MCU access window into 2 */ -	ipu_cm_write(ipu, 0x00400000L | (IPU_MCU_T_DEFAULT << 18), -			IPU_DISP_GEN); - -	ret = ipu_submodules_init(ipu, pdev, ipu_base, ipu->clk); -	if (ret) -		goto failed_submodules_init; - -	ret = ipu_add_client_devices(ipu); -	if (ret) { -		dev_err(&pdev->dev, "adding client devices failed with %d\n", -				ret); -		goto failed_add_clients; -	} - -	dev_info(&pdev->dev, "%s probed\n", devtype->name); - -	return 0; - -failed_add_clients: -	ipu_submodules_exit(ipu); -failed_submodules_init: -out_failed_reset: -	ipu_irq_exit(ipu); -out_failed_irq: -	clk_disable_unprepare(ipu->clk); -	return ret; -} - -static int ipu_remove(struct platform_device *pdev) -{ -	struct ipu_soc *ipu = platform_get_drvdata(pdev); - -	platform_device_unregister_children(pdev); -	ipu_submodules_exit(ipu); -	ipu_irq_exit(ipu); - -	clk_disable_unprepare(ipu->clk); - -	return 0; -} - -static struct platform_driver imx_ipu_driver = { -	.driver = { -		.name = "imx-ipuv3", -		.of_match_table = imx_ipu_dt_ids, -	}, -	.probe = ipu_probe, -	.remove = ipu_remove, -}; - -module_platform_driver(imx_ipu_driver); - -MODULE_ALIAS("platform:imx-ipuv3"); -MODULE_DESCRIPTION("i.MX IPU v3 driver"); -MODULE_AUTHOR("Sascha Hauer <s.hauer@pengutronix.de>"); -MODULE_LICENSE("GPL"); diff --git a/drivers/staging/imx-drm/ipu-v3/ipu-dc.c b/drivers/staging/imx-drm/ipu-v3/ipu-dc.c deleted file mode 100644 index 21bf1c80652..00000000000 --- a/drivers/staging/imx-drm/ipu-v3/ipu-dc.c +++ /dev/null @@ -1,403 +0,0 @@ -/* - * Copyright (c) 2010 Sascha Hauer <s.hauer@pengutronix.de> - * Copyright (C) 2005-2009 Freescale Semiconductor, Inc. - * - * This program is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License as published by the - * Free Software Foundation; either version 2 of the License, or (at your - * option) any later version. - * - * This program is distributed in the hope that it will be useful, but - * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY - * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License - * for more details. - */ - -#include <linux/export.h> -#include <linux/module.h> -#include <linux/types.h> -#include <linux/errno.h> -#include <linux/delay.h> -#include <linux/io.h> - -#include "../imx-drm.h" -#include "imx-ipu-v3.h" -#include "ipu-prv.h" - -#define DC_MAP_CONF_PTR(n)	(0x108 + ((n) & ~0x1) * 2) -#define DC_MAP_CONF_VAL(n)	(0x144 + ((n) & ~0x1) * 2) - -#define DC_EVT_NF		0 -#define DC_EVT_NL		1 -#define DC_EVT_EOF		2 -#define DC_EVT_NFIELD		3 -#define DC_EVT_EOL		4 -#define DC_EVT_EOFIELD		5 -#define DC_EVT_NEW_ADDR		6 -#define DC_EVT_NEW_CHAN		7 -#define DC_EVT_NEW_DATA		8 - -#define DC_EVT_NEW_ADDR_W_0	0 -#define DC_EVT_NEW_ADDR_W_1	1 -#define DC_EVT_NEW_CHAN_W_0	2 -#define DC_EVT_NEW_CHAN_W_1	3 -#define DC_EVT_NEW_DATA_W_0	4 -#define DC_EVT_NEW_DATA_W_1	5 -#define DC_EVT_NEW_ADDR_R_0	6 -#define DC_EVT_NEW_ADDR_R_1	7 -#define DC_EVT_NEW_CHAN_R_0	8 -#define DC_EVT_NEW_CHAN_R_1	9 -#define DC_EVT_NEW_DATA_R_0	10 -#define DC_EVT_NEW_DATA_R_1	11 - -#define DC_WR_CH_CONF		0x0 -#define DC_WR_CH_ADDR		0x4 -#define DC_RL_CH(evt)		(8 + ((evt) & ~0x1) * 2) - -#define DC_GEN			0xd4 -#define DC_DISP_CONF1(disp)	(0xd8 + (disp) * 4) -#define DC_DISP_CONF2(disp)	(0xe8 + (disp) * 4) -#define DC_STAT			0x1c8 - -#define WROD(lf)		(0x18 | ((lf) << 1)) -#define WRG			0x01 -#define WCLK			0xc9 - -#define SYNC_WAVE 0 -#define NULL_WAVE (-1) - -#define DC_GEN_SYNC_1_6_SYNC	(2 << 1) -#define DC_GEN_SYNC_PRIORITY_1	(1 << 7) - -#define DC_WR_CH_CONF_WORD_SIZE_8		(0 << 0) -#define DC_WR_CH_CONF_WORD_SIZE_16		(1 << 0) -#define DC_WR_CH_CONF_WORD_SIZE_24		(2 << 0) -#define DC_WR_CH_CONF_WORD_SIZE_32		(3 << 0) -#define DC_WR_CH_CONF_DISP_ID_PARALLEL(i)	(((i) & 0x1) << 3) -#define DC_WR_CH_CONF_DISP_ID_SERIAL		(2 << 3) -#define DC_WR_CH_CONF_DISP_ID_ASYNC		(3 << 4) -#define DC_WR_CH_CONF_FIELD_MODE		(1 << 9) -#define DC_WR_CH_CONF_PROG_TYPE_NORMAL		(4 << 5) -#define DC_WR_CH_CONF_PROG_TYPE_MASK		(7 << 5) -#define DC_WR_CH_CONF_PROG_DI_ID		(1 << 2) -#define DC_WR_CH_CONF_PROG_DISP_ID(i)		(((i) & 0x1) << 3) - -#define IPU_DC_NUM_CHANNELS	10 - -struct ipu_dc_priv; - -enum ipu_dc_map { -	IPU_DC_MAP_RGB24, -	IPU_DC_MAP_RGB565, -	IPU_DC_MAP_GBR24, /* TVEv2 */ -	IPU_DC_MAP_BGR666, -}; - -struct ipu_dc { -	/* The display interface number assigned to this dc channel */ -	unsigned int		di; -	void __iomem		*base; -	struct ipu_dc_priv	*priv; -	int			chno; -	bool			in_use; -}; - -struct ipu_dc_priv { -	void __iomem		*dc_reg; -	void __iomem		*dc_tmpl_reg; -	struct ipu_soc		*ipu; -	struct device		*dev; -	struct ipu_dc		channels[IPU_DC_NUM_CHANNELS]; -	struct mutex		mutex; -}; - -static void dc_link_event(struct ipu_dc *dc, int event, int addr, int priority) -{ -	u32 reg; - -	reg = readl(dc->base + DC_RL_CH(event)); -	reg &= ~(0xffff << (16 * (event & 0x1))); -	reg |= ((addr << 8) | priority) << (16 * (event & 0x1)); -	writel(reg, dc->base + DC_RL_CH(event)); -} - -static void dc_write_tmpl(struct ipu_dc *dc, int word, u32 opcode, u32 operand, -		int map, int wave, int glue, int sync, int stop) -{ -	struct ipu_dc_priv *priv = dc->priv; -	u32 reg1, reg2; - -	if (opcode == WCLK) { -		reg1 = (operand << 20) & 0xfff00000; -		reg2 = operand >> 12 | opcode << 1 | stop << 9; -	} else if (opcode == WRG) { -		reg1 = sync | glue << 4 | ++wave << 11 | ((operand << 15) & 0xffff8000); -		reg2 = operand >> 17 | opcode << 7 | stop << 9; -	} else { -		reg1 = sync | glue << 4 | ++wave << 11 | ++map << 15 | ((operand << 20) & 0xfff00000); -		reg2 = operand >> 12 | opcode << 4 | stop << 9; -	} -	writel(reg1, priv->dc_tmpl_reg + word * 8); -	writel(reg2, priv->dc_tmpl_reg + word * 8 + 4); -} - -static int ipu_pixfmt_to_map(u32 fmt) -{ -	switch (fmt) { -	case V4L2_PIX_FMT_RGB24: -		return IPU_DC_MAP_RGB24; -	case V4L2_PIX_FMT_RGB565: -		return IPU_DC_MAP_RGB565; -	case IPU_PIX_FMT_GBR24: -		return IPU_DC_MAP_GBR24; -	case V4L2_PIX_FMT_BGR666: -		return IPU_DC_MAP_BGR666; -	default: -		return -EINVAL; -	} -} - -int ipu_dc_init_sync(struct ipu_dc *dc, struct ipu_di *di, bool interlaced, -		u32 pixel_fmt, u32 width) -{ -	struct ipu_dc_priv *priv = dc->priv; -	u32 reg = 0; -	int map; - -	dc->di = ipu_di_get_num(di); - -	map = ipu_pixfmt_to_map(pixel_fmt); -	if (map < 0) { -		dev_dbg(priv->dev, "IPU_DISP: No MAP\n"); -		return map; -	} - -	if (interlaced) { -		dc_link_event(dc, DC_EVT_NL, 0, 3); -		dc_link_event(dc, DC_EVT_EOL, 0, 2); -		dc_link_event(dc, DC_EVT_NEW_DATA, 0, 1); - -		/* Init template microcode */ -		dc_write_tmpl(dc, 0, WROD(0), 0, map, SYNC_WAVE, 0, 8, 1); -	} else { -		if (dc->di) { -			dc_link_event(dc, DC_EVT_NL, 2, 3); -			dc_link_event(dc, DC_EVT_EOL, 3, 2); -			dc_link_event(dc, DC_EVT_NEW_DATA, 1, 1); -			/* Init template microcode */ -			dc_write_tmpl(dc, 2, WROD(0), 0, map, SYNC_WAVE, 8, 5, 1); -			dc_write_tmpl(dc, 3, WROD(0), 0, map, SYNC_WAVE, 4, 5, 0); -			dc_write_tmpl(dc, 4, WRG, 0, map, NULL_WAVE, 0, 0, 1); -			dc_write_tmpl(dc, 1, WROD(0), 0, map, SYNC_WAVE, 0, 5, 1); -		} else { -			dc_link_event(dc, DC_EVT_NL, 5, 3); -			dc_link_event(dc, DC_EVT_EOL, 6, 2); -			dc_link_event(dc, DC_EVT_NEW_DATA, 8, 1); -			/* Init template microcode */ -			dc_write_tmpl(dc, 5, WROD(0), 0, map, SYNC_WAVE, 8, 5, 1); -			dc_write_tmpl(dc, 6, WROD(0), 0, map, SYNC_WAVE, 4, 5, 0); -			dc_write_tmpl(dc, 7, WRG, 0, map, NULL_WAVE, 0, 0, 1); -			dc_write_tmpl(dc, 8, WROD(0), 0, map, SYNC_WAVE, 0, 5, 1); -		} -	} -	dc_link_event(dc, DC_EVT_NF, 0, 0); -	dc_link_event(dc, DC_EVT_NFIELD, 0, 0); -	dc_link_event(dc, DC_EVT_EOF, 0, 0); -	dc_link_event(dc, DC_EVT_EOFIELD, 0, 0); -	dc_link_event(dc, DC_EVT_NEW_CHAN, 0, 0); -	dc_link_event(dc, DC_EVT_NEW_ADDR, 0, 0); - -	reg = readl(dc->base + DC_WR_CH_CONF); -	if (interlaced) -		reg |= DC_WR_CH_CONF_FIELD_MODE; -	else -		reg &= ~DC_WR_CH_CONF_FIELD_MODE; -	writel(reg, dc->base + DC_WR_CH_CONF); - -	writel(0x0, dc->base + DC_WR_CH_ADDR); -	writel(width, priv->dc_reg + DC_DISP_CONF2(dc->di)); - -	ipu_module_enable(priv->ipu, IPU_CONF_DC_EN); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_dc_init_sync); - -void ipu_dc_enable_channel(struct ipu_dc *dc) -{ -	int di; -	u32 reg; - -	di = dc->di; - -	reg = readl(dc->base + DC_WR_CH_CONF); -	reg |= DC_WR_CH_CONF_PROG_TYPE_NORMAL; -	writel(reg, dc->base + DC_WR_CH_CONF); -} -EXPORT_SYMBOL_GPL(ipu_dc_enable_channel); - -void ipu_dc_disable_channel(struct ipu_dc *dc) -{ -	struct ipu_dc_priv *priv = dc->priv; -	u32 val; -	int irq = 0, timeout = 50; - -	if (dc->chno == 1) -		irq = IPU_IRQ_DC_FC_1; -	else if (dc->chno == 5) -		irq = IPU_IRQ_DP_SF_END; -	else -		return; - -	/* should wait for the interrupt here */ -	mdelay(50); - -	if (dc->di == 0) -		val = 0x00000002; -	else -		val = 0x00000020; - -	/* Wait for DC triple buffer to empty */ -	while ((readl(priv->dc_reg + DC_STAT) & val) != val) { -		msleep(2); -		timeout -= 2; -		if (timeout <= 0) -			break; -	} - -	val = readl(dc->base + DC_WR_CH_CONF); -	val &= ~DC_WR_CH_CONF_PROG_TYPE_MASK; -	writel(val, dc->base + DC_WR_CH_CONF); -} -EXPORT_SYMBOL_GPL(ipu_dc_disable_channel); - -static void ipu_dc_map_config(struct ipu_dc_priv *priv, enum ipu_dc_map map, -		int byte_num, int offset, int mask) -{ -	int ptr = map * 3 + byte_num; -	u32 reg; - -	reg = readl(priv->dc_reg + DC_MAP_CONF_VAL(ptr)); -	reg &= ~(0xffff << (16 * (ptr & 0x1))); -	reg |= ((offset << 8) | mask) << (16 * (ptr & 0x1)); -	writel(reg, priv->dc_reg + DC_MAP_CONF_VAL(ptr)); - -	reg = readl(priv->dc_reg + DC_MAP_CONF_PTR(map)); -	reg &= ~(0x1f << ((16 * (map & 0x1)) + (5 * byte_num))); -	reg |= ptr << ((16 * (map & 0x1)) + (5 * byte_num)); -	writel(reg, priv->dc_reg + DC_MAP_CONF_PTR(map)); -} - -static void ipu_dc_map_clear(struct ipu_dc_priv *priv, int map) -{ -	u32 reg = readl(priv->dc_reg + DC_MAP_CONF_PTR(map)); - -	writel(reg & ~(0xffff << (16 * (map & 0x1))), -		     priv->dc_reg + DC_MAP_CONF_PTR(map)); -} - -struct ipu_dc *ipu_dc_get(struct ipu_soc *ipu, int channel) -{ -	struct ipu_dc_priv *priv = ipu->dc_priv; -	struct ipu_dc *dc; - -	if (channel >= IPU_DC_NUM_CHANNELS) -		return ERR_PTR(-ENODEV); - -	dc = &priv->channels[channel]; - -	mutex_lock(&priv->mutex); - -	if (dc->in_use) { -		mutex_unlock(&priv->mutex); -		return ERR_PTR(-EBUSY); -	} - -	dc->in_use = 1; - -	mutex_unlock(&priv->mutex); - -	return dc; -} -EXPORT_SYMBOL_GPL(ipu_dc_get); - -void ipu_dc_put(struct ipu_dc *dc) -{ -	struct ipu_dc_priv *priv = dc->priv; - -	mutex_lock(&priv->mutex); -	dc->in_use = 0; -	mutex_unlock(&priv->mutex); -} -EXPORT_SYMBOL_GPL(ipu_dc_put); - -int ipu_dc_init(struct ipu_soc *ipu, struct device *dev, -		unsigned long base, unsigned long template_base) -{ -	struct ipu_dc_priv *priv; -	static int channel_offsets[] = { 0, 0x1c, 0x38, 0x54, 0x58, 0x5c, -		0x78, 0, 0x94, 0xb4}; -	int i; - -	priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); -	if (!priv) -		return -ENOMEM; - -	mutex_init(&priv->mutex); - -	priv->dev = dev; -	priv->ipu = ipu; -	priv->dc_reg = devm_ioremap(dev, base, PAGE_SIZE); -	priv->dc_tmpl_reg = devm_ioremap(dev, template_base, PAGE_SIZE); -	if (!priv->dc_reg || !priv->dc_tmpl_reg) -		return -ENOMEM; - -	for (i = 0; i < IPU_DC_NUM_CHANNELS; i++) { -		priv->channels[i].chno = i; -		priv->channels[i].priv = priv; -		priv->channels[i].base = priv->dc_reg + channel_offsets[i]; -	} - -	writel(DC_WR_CH_CONF_WORD_SIZE_24 | DC_WR_CH_CONF_DISP_ID_PARALLEL(1) | -			DC_WR_CH_CONF_PROG_DI_ID, -			priv->channels[1].base + DC_WR_CH_CONF); -	writel(DC_WR_CH_CONF_WORD_SIZE_24 | DC_WR_CH_CONF_DISP_ID_PARALLEL(0), -			priv->channels[5].base + DC_WR_CH_CONF); - -	writel(DC_GEN_SYNC_1_6_SYNC | DC_GEN_SYNC_PRIORITY_1, priv->dc_reg + DC_GEN); - -	ipu->dc_priv = priv; - -	dev_dbg(dev, "DC base: 0x%08lx template base: 0x%08lx\n", -			base, template_base); - -	/* rgb24 */ -	ipu_dc_map_clear(priv, IPU_DC_MAP_RGB24); -	ipu_dc_map_config(priv, IPU_DC_MAP_RGB24, 0, 7, 0xff); /* blue */ -	ipu_dc_map_config(priv, IPU_DC_MAP_RGB24, 1, 15, 0xff); /* green */ -	ipu_dc_map_config(priv, IPU_DC_MAP_RGB24, 2, 23, 0xff); /* red */ - -	/* rgb565 */ -	ipu_dc_map_clear(priv, IPU_DC_MAP_RGB565); -	ipu_dc_map_config(priv, IPU_DC_MAP_RGB565, 0, 4, 0xf8); /* blue */ -	ipu_dc_map_config(priv, IPU_DC_MAP_RGB565, 1, 10, 0xfc); /* green */ -	ipu_dc_map_config(priv, IPU_DC_MAP_RGB565, 2, 15, 0xf8); /* red */ - -	/* gbr24 */ -	ipu_dc_map_clear(priv, IPU_DC_MAP_GBR24); -	ipu_dc_map_config(priv, IPU_DC_MAP_GBR24, 2, 15, 0xff); /* green */ -	ipu_dc_map_config(priv, IPU_DC_MAP_GBR24, 1, 7, 0xff); /* blue */ -	ipu_dc_map_config(priv, IPU_DC_MAP_GBR24, 0, 23, 0xff); /* red */ - -	/* bgr666 */ -	ipu_dc_map_clear(priv, IPU_DC_MAP_BGR666); -	ipu_dc_map_config(priv, IPU_DC_MAP_BGR666, 0, 5, 0xfc); /* blue */ -	ipu_dc_map_config(priv, IPU_DC_MAP_BGR666, 1, 11, 0xfc); /* green */ -	ipu_dc_map_config(priv, IPU_DC_MAP_BGR666, 2, 17, 0xfc); /* red */ - -	return 0; -} - -void ipu_dc_exit(struct ipu_soc *ipu) -{ -} diff --git a/drivers/staging/imx-drm/ipu-v3/ipu-di.c b/drivers/staging/imx-drm/ipu-v3/ipu-di.c deleted file mode 100644 index 948a49b289e..00000000000 --- a/drivers/staging/imx-drm/ipu-v3/ipu-di.c +++ /dev/null @@ -1,793 +0,0 @@ -/* - * Copyright (c) 2010 Sascha Hauer <s.hauer@pengutronix.de> - * Copyright (C) 2005-2009 Freescale Semiconductor, Inc. - * - * This program is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License as published by the - * Free Software Foundation; either version 2 of the License, or (at your - * option) any later version. - * - * This program is distributed in the hope that it will be useful, but - * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY - * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License - * for more details. - */ -#include <linux/export.h> -#include <linux/module.h> -#include <linux/types.h> -#include <linux/errno.h> -#include <linux/io.h> -#include <linux/err.h> -#include <linux/platform_device.h> -#include <linux/clk.h> -#include <linux/clk-provider.h> -#include <linux/clkdev.h> - -#include "imx-ipu-v3.h" -#include "ipu-prv.h" - -struct ipu_di { -	void __iomem *base; -	int id; -	u32 module; -	struct clk *clk_di;	/* display input clock */ -	struct clk *clk_ipu;	/* IPU bus clock */ -	struct clk *clk_di_pixel; /* resulting pixel clock */ -	struct clk_hw clk_hw_out; -	char *clk_name; -	bool inuse; -	unsigned long clkflags; -	struct ipu_soc *ipu; -}; - -static DEFINE_MUTEX(di_mutex); - -struct di_sync_config { -	int run_count; -	int run_src; -	int offset_count; -	int offset_src; -	int repeat_count; -	int cnt_clr_src; -	int cnt_polarity_gen_en; -	int cnt_polarity_clr_src; -	int cnt_polarity_trigger_src; -	int cnt_up; -	int cnt_down; -}; - -enum di_pins { -	DI_PIN11 = 0, -	DI_PIN12 = 1, -	DI_PIN13 = 2, -	DI_PIN14 = 3, -	DI_PIN15 = 4, -	DI_PIN16 = 5, -	DI_PIN17 = 6, -	DI_PIN_CS = 7, - -	DI_PIN_SER_CLK = 0, -	DI_PIN_SER_RS = 1, -}; - -enum di_sync_wave { -	DI_SYNC_NONE = 0, -	DI_SYNC_CLK = 1, -	DI_SYNC_INT_HSYNC = 2, -	DI_SYNC_HSYNC = 3, -	DI_SYNC_VSYNC = 4, -	DI_SYNC_DE = 6, -}; - -#define SYNC_WAVE 0 - -#define DI_GENERAL		0x0000 -#define DI_BS_CLKGEN0		0x0004 -#define DI_BS_CLKGEN1		0x0008 -#define DI_SW_GEN0(gen)		(0x000c + 4 * ((gen) - 1)) -#define DI_SW_GEN1(gen)		(0x0030 + 4 * ((gen) - 1)) -#define DI_STP_REP(gen)		(0x0148 + 4 * (((gen) - 1)/2)) -#define DI_SYNC_AS_GEN		0x0054 -#define DI_DW_GEN(gen)		(0x0058 + 4 * (gen)) -#define DI_DW_SET(gen, set)	(0x0088 + 4 * ((gen) + 0xc * (set))) -#define DI_SER_CONF		0x015c -#define DI_SSC			0x0160 -#define DI_POL			0x0164 -#define DI_AW0			0x0168 -#define DI_AW1			0x016c -#define DI_SCR_CONF		0x0170 -#define DI_STAT			0x0174 - -#define DI_SW_GEN0_RUN_COUNT(x)			((x) << 19) -#define DI_SW_GEN0_RUN_SRC(x)			((x) << 16) -#define DI_SW_GEN0_OFFSET_COUNT(x)		((x) << 3) -#define DI_SW_GEN0_OFFSET_SRC(x)		((x) << 0) - -#define DI_SW_GEN1_CNT_POL_GEN_EN(x)		((x) << 29) -#define DI_SW_GEN1_CNT_CLR_SRC(x)		((x) << 25) -#define DI_SW_GEN1_CNT_POL_TRIGGER_SRC(x)	((x) << 12) -#define DI_SW_GEN1_CNT_POL_CLR_SRC(x)		((x) << 9) -#define DI_SW_GEN1_CNT_DOWN(x)			((x) << 16) -#define DI_SW_GEN1_CNT_UP(x)			(x) -#define DI_SW_GEN1_AUTO_RELOAD			(0x10000000) - -#define DI_DW_GEN_ACCESS_SIZE_OFFSET		24 -#define DI_DW_GEN_COMPONENT_SIZE_OFFSET		16 - -#define DI_GEN_POLARITY_1			(1 << 0) -#define DI_GEN_POLARITY_2			(1 << 1) -#define DI_GEN_POLARITY_3			(1 << 2) -#define DI_GEN_POLARITY_4			(1 << 3) -#define DI_GEN_POLARITY_5			(1 << 4) -#define DI_GEN_POLARITY_6			(1 << 5) -#define DI_GEN_POLARITY_7			(1 << 6) -#define DI_GEN_POLARITY_8			(1 << 7) -#define DI_GEN_POLARITY_DISP_CLK		(1 << 17) -#define DI_GEN_DI_CLK_EXT			(1 << 20) -#define DI_GEN_DI_VSYNC_EXT			(1 << 21) - -#define DI_POL_DRDY_DATA_POLARITY		(1 << 7) -#define DI_POL_DRDY_POLARITY_15			(1 << 4) - -#define DI_VSYNC_SEL_OFFSET			13 - -static inline u32 ipu_di_read(struct ipu_di *di, unsigned offset) -{ -	return readl(di->base + offset); -} - -static inline void ipu_di_write(struct ipu_di *di, u32 value, unsigned offset) -{ -	writel(value, di->base + offset); -} - -static int ipu_di_clk_calc_div(unsigned long inrate, unsigned long outrate) -{ -	u64 tmp = inrate; -	int div; - -	tmp *= 16; - -	do_div(tmp, outrate); - -	div = tmp; - -	if (div < 0x10) -		div = 0x10; - -#ifdef WTF_IS_THIS -	/* -	 * Freescale has this in their Kernel. It is neither clear what -	 * it does nor why it does it -	 */ -	if (div & 0x10) -		div &= ~0x7; -	else { -		/* Round up divider if it gets us closer to desired pix clk */ -		if ((div & 0xC) == 0xC) { -			div += 0x10; -			div &= ~0xF; -		} -	} -#endif -	return div; -} - -static unsigned long clk_di_recalc_rate(struct clk_hw *hw, -		unsigned long parent_rate) -{ -	struct ipu_di *di = container_of(hw, struct ipu_di, clk_hw_out); -	unsigned long outrate; -	u32 div = ipu_di_read(di, DI_BS_CLKGEN0); - -	if (div < 0x10) -		div = 0x10; - -	outrate = (parent_rate / div) * 16; - -	return outrate; -} - -static long clk_di_round_rate(struct clk_hw *hw, unsigned long rate, -				unsigned long *prate) -{ -	struct ipu_di *di = container_of(hw, struct ipu_di, clk_hw_out); -	unsigned long outrate; -	int div; -	u32 val; - -	div = ipu_di_clk_calc_div(*prate, rate); - -	outrate = (*prate / div) * 16; - -	val = ipu_di_read(di, DI_GENERAL); - -	if (!(val & DI_GEN_DI_CLK_EXT) && outrate > *prate / 2) -		outrate = *prate / 2; - -	dev_dbg(di->ipu->dev, -		"%s: inrate: %ld div: 0x%08x outrate: %ld wanted: %ld\n", -			__func__, *prate, div, outrate, rate); - -	return outrate; -} - -static int clk_di_set_rate(struct clk_hw *hw, unsigned long rate, -				unsigned long parent_rate) -{ -	struct ipu_di *di = container_of(hw, struct ipu_di, clk_hw_out); -	int div; -	u32 clkgen0; - -	clkgen0 = ipu_di_read(di, DI_BS_CLKGEN0) & ~0xfff; - -	div = ipu_di_clk_calc_div(parent_rate, rate); - -	ipu_di_write(di, clkgen0 | div, DI_BS_CLKGEN0); - -	dev_dbg(di->ipu->dev, "%s: inrate: %ld desired: %ld div: 0x%08x\n", -			__func__, parent_rate, rate, div); -	return 0; -} - -static u8 clk_di_get_parent(struct clk_hw *hw) -{ -	struct ipu_di *di = container_of(hw, struct ipu_di, clk_hw_out); -	u32 val; - -	val = ipu_di_read(di, DI_GENERAL); - -	return val & DI_GEN_DI_CLK_EXT ? 1 : 0; -} - -static int clk_di_set_parent(struct clk_hw *hw, u8 index) -{ -	struct ipu_di *di = container_of(hw, struct ipu_di, clk_hw_out); -	u32 val; - -	val = ipu_di_read(di, DI_GENERAL); - -	if (index) -		val |= DI_GEN_DI_CLK_EXT; -	else -		val &= ~DI_GEN_DI_CLK_EXT; - -	ipu_di_write(di, val, DI_GENERAL); - -	return 0; -} - -static struct clk_ops clk_di_ops = { -	.round_rate = clk_di_round_rate, -	.set_rate = clk_di_set_rate, -	.recalc_rate = clk_di_recalc_rate, -	.set_parent = clk_di_set_parent, -	.get_parent = clk_di_get_parent, -}; - -static void ipu_di_data_wave_config(struct ipu_di *di, -				     int wave_gen, -				     int access_size, int component_size) -{ -	u32 reg; -	reg = (access_size << DI_DW_GEN_ACCESS_SIZE_OFFSET) | -	    (component_size << DI_DW_GEN_COMPONENT_SIZE_OFFSET); -	ipu_di_write(di, reg, DI_DW_GEN(wave_gen)); -} - -static void ipu_di_data_pin_config(struct ipu_di *di, int wave_gen, int di_pin, -		int set, int up, int down) -{ -	u32 reg; - -	reg = ipu_di_read(di, DI_DW_GEN(wave_gen)); -	reg &= ~(0x3 << (di_pin * 2)); -	reg |= set << (di_pin * 2); -	ipu_di_write(di, reg, DI_DW_GEN(wave_gen)); - -	ipu_di_write(di, (down << 16) | up, DI_DW_SET(wave_gen, set)); -} - -static void ipu_di_sync_config(struct ipu_di *di, struct di_sync_config *config, -		int start, int count) -{ -	u32 reg; -	int i; - -	for (i = 0; i < count; i++) { -		struct di_sync_config *c = &config[i]; -		int wave_gen = start + i + 1; - -		if ((c->run_count >= 0x1000) || (c->offset_count >= 0x1000) || -				(c->repeat_count >= 0x1000) || -				(c->cnt_up >= 0x400) || -				(c->cnt_down >= 0x400)) { -			dev_err(di->ipu->dev, "DI%d counters out of range.\n", -					di->id); -			return; -		} - -		reg = DI_SW_GEN0_RUN_COUNT(c->run_count) | -			DI_SW_GEN0_RUN_SRC(c->run_src) | -			DI_SW_GEN0_OFFSET_COUNT(c->offset_count) | -			DI_SW_GEN0_OFFSET_SRC(c->offset_src); -		ipu_di_write(di, reg, DI_SW_GEN0(wave_gen)); - -		reg = DI_SW_GEN1_CNT_POL_GEN_EN(c->cnt_polarity_gen_en) | -			DI_SW_GEN1_CNT_CLR_SRC(c->cnt_clr_src) | -			DI_SW_GEN1_CNT_POL_TRIGGER_SRC( -					c->cnt_polarity_trigger_src) | -			DI_SW_GEN1_CNT_POL_CLR_SRC(c->cnt_polarity_clr_src) | -			DI_SW_GEN1_CNT_DOWN(c->cnt_down) | -			DI_SW_GEN1_CNT_UP(c->cnt_up); - -		/* Enable auto reload */ -		if (c->repeat_count == 0) -			reg |= DI_SW_GEN1_AUTO_RELOAD; - -		ipu_di_write(di, reg, DI_SW_GEN1(wave_gen)); - -		reg = ipu_di_read(di, DI_STP_REP(wave_gen)); -		reg &= ~(0xffff << (16 * ((wave_gen - 1) & 0x1))); -		reg |= c->repeat_count << (16 * ((wave_gen - 1) & 0x1)); -		ipu_di_write(di, reg, DI_STP_REP(wave_gen)); -	} -} - -static void ipu_di_sync_config_interlaced(struct ipu_di *di, -		struct ipu_di_signal_cfg *sig) -{ -	u32 h_total = sig->width + sig->h_sync_width + -		sig->h_start_width + sig->h_end_width; -	u32 v_total = sig->height + sig->v_sync_width + -		sig->v_start_width + sig->v_end_width; -	u32 reg; -	struct di_sync_config cfg[] = { -		{ -			.run_count = h_total / 2 - 1, -			.run_src = DI_SYNC_CLK, -		}, { -			.run_count = h_total - 11, -			.run_src = DI_SYNC_CLK, -			.cnt_down = 4, -		}, { -			.run_count = v_total * 2 - 1, -			.run_src = DI_SYNC_INT_HSYNC, -			.offset_count = 1, -			.offset_src = DI_SYNC_INT_HSYNC, -			.cnt_down = 4, -		}, { -			.run_count = v_total / 2 - 1, -			.run_src = DI_SYNC_HSYNC, -			.offset_count = sig->v_start_width, -			.offset_src = DI_SYNC_HSYNC, -			.repeat_count = 2, -			.cnt_clr_src = DI_SYNC_VSYNC, -		}, { -			.run_src = DI_SYNC_HSYNC, -			.repeat_count = sig->height / 2, -			.cnt_clr_src = 4, -		}, { -			.run_count = v_total - 1, -			.run_src = DI_SYNC_HSYNC, -		}, { -			.run_count = v_total / 2 - 1, -			.run_src = DI_SYNC_HSYNC, -			.offset_count = 9, -			.offset_src = DI_SYNC_HSYNC, -			.repeat_count = 2, -			.cnt_clr_src = DI_SYNC_VSYNC, -		}, { -			.run_src = DI_SYNC_CLK, -			.offset_count = sig->h_start_width, -			.offset_src = DI_SYNC_CLK, -			.repeat_count = sig->width, -			.cnt_clr_src = 5, -		}, { -			.run_count = v_total - 1, -			.run_src = DI_SYNC_INT_HSYNC, -			.offset_count = v_total / 2, -			.offset_src = DI_SYNC_INT_HSYNC, -			.cnt_clr_src = DI_SYNC_HSYNC, -			.cnt_down = 4, -		} -	}; - -	ipu_di_sync_config(di, cfg, 0, ARRAY_SIZE(cfg)); - -	/* set gentime select and tag sel */ -	reg = ipu_di_read(di, DI_SW_GEN1(9)); -	reg &= 0x1FFFFFFF; -	reg |= (3 - 1) << 29 | 0x00008000; -	ipu_di_write(di, reg, DI_SW_GEN1(9)); - -	ipu_di_write(di, v_total / 2 - 1, DI_SCR_CONF); -} - -static void ipu_di_sync_config_noninterlaced(struct ipu_di *di, -		struct ipu_di_signal_cfg *sig, int div) -{ -	u32 h_total = sig->width + sig->h_sync_width + sig->h_start_width + -		sig->h_end_width; -	u32 v_total = sig->height + sig->v_sync_width + sig->v_start_width + -		sig->v_end_width; -	struct di_sync_config cfg[] = { -		{ -			/* 1: INT_HSYNC */ -			.run_count = h_total - 1, -			.run_src = DI_SYNC_CLK, -		} , { -			/* PIN2: HSYNC */ -			.run_count = h_total - 1, -			.run_src = DI_SYNC_CLK, -			.offset_count = div * sig->v_to_h_sync, -			.offset_src = DI_SYNC_CLK, -			.cnt_polarity_gen_en = 1, -			.cnt_polarity_trigger_src = DI_SYNC_CLK, -			.cnt_down = sig->h_sync_width * 2, -		} , { -			/* PIN3: VSYNC */ -			.run_count = v_total - 1, -			.run_src = DI_SYNC_INT_HSYNC, -			.cnt_polarity_gen_en = 1, -			.cnt_polarity_trigger_src = DI_SYNC_INT_HSYNC, -			.cnt_down = sig->v_sync_width * 2, -		} , { -			/* 4: Line Active */ -			.run_src = DI_SYNC_HSYNC, -			.offset_count = sig->v_sync_width + sig->v_start_width, -			.offset_src = DI_SYNC_HSYNC, -			.repeat_count = sig->height, -			.cnt_clr_src = DI_SYNC_VSYNC, -		} , { -			/* 5: Pixel Active, referenced by DC */ -			.run_src = DI_SYNC_CLK, -			.offset_count = sig->h_sync_width + sig->h_start_width, -			.offset_src = DI_SYNC_CLK, -			.repeat_count = sig->width, -			.cnt_clr_src = 5, /* Line Active */ -		} , { -			/* unused */ -		} , { -			/* unused */ -		} , { -			/* unused */ -		} , { -			/* unused */ -		}, -	}; -	/* can't use #7 and #8 for line active and pixel active counters */ -	struct di_sync_config cfg_vga[] = { -		{ -			/* 1: INT_HSYNC */ -			.run_count = h_total - 1, -			.run_src = DI_SYNC_CLK, -		} , { -			/* 2: VSYNC */ -			.run_count = v_total - 1, -			.run_src = DI_SYNC_INT_HSYNC, -		} , { -			/* 3: Line Active */ -			.run_src = DI_SYNC_INT_HSYNC, -			.offset_count = sig->v_sync_width + sig->v_start_width, -			.offset_src = DI_SYNC_INT_HSYNC, -			.repeat_count = sig->height, -			.cnt_clr_src = 3 /* VSYNC */, -		} , { -			/* PIN4: HSYNC for VGA via TVEv2 on TQ MBa53 */ -			.run_count = h_total - 1, -			.run_src = DI_SYNC_CLK, -			.offset_count = div * sig->v_to_h_sync + 18, /* magic value from Freescale TVE driver */ -			.offset_src = DI_SYNC_CLK, -			.cnt_polarity_gen_en = 1, -			.cnt_polarity_trigger_src = DI_SYNC_CLK, -			.cnt_down = sig->h_sync_width * 2, -		} , { -			/* 5: Pixel Active signal to DC */ -			.run_src = DI_SYNC_CLK, -			.offset_count = sig->h_sync_width + sig->h_start_width, -			.offset_src = DI_SYNC_CLK, -			.repeat_count = sig->width, -			.cnt_clr_src = 4, /* Line Active */ -		} , { -			/* PIN6: VSYNC for VGA via TVEv2 on TQ MBa53 */ -			.run_count = v_total - 1, -			.run_src = DI_SYNC_INT_HSYNC, -			.offset_count = 1, /* magic value from Freescale TVE driver */ -			.offset_src = DI_SYNC_INT_HSYNC, -			.cnt_polarity_gen_en = 1, -			.cnt_polarity_trigger_src = DI_SYNC_INT_HSYNC, -			.cnt_down = sig->v_sync_width * 2, -		} , { -			/* PIN4: HSYNC for VGA via TVEv2 on i.MX53-QSB */ -			.run_count = h_total - 1, -			.run_src = DI_SYNC_CLK, -			.offset_count = div * sig->v_to_h_sync + 18, /* magic value from Freescale TVE driver */ -			.offset_src = DI_SYNC_CLK, -			.cnt_polarity_gen_en = 1, -			.cnt_polarity_trigger_src = DI_SYNC_CLK, -			.cnt_down = sig->h_sync_width * 2, -		} , { -			/* PIN6: VSYNC for VGA via TVEv2 on i.MX53-QSB */ -			.run_count = v_total - 1, -			.run_src = DI_SYNC_INT_HSYNC, -			.offset_count = 1, /* magic value from Freescale TVE driver */ -			.offset_src = DI_SYNC_INT_HSYNC, -			.cnt_polarity_gen_en = 1, -			.cnt_polarity_trigger_src = DI_SYNC_INT_HSYNC, -			.cnt_down = sig->v_sync_width * 2, -		} , { -			/* unused */ -		}, -	}; - -	ipu_di_write(di, v_total - 1, DI_SCR_CONF); -	if (sig->hsync_pin == 2 && sig->vsync_pin == 3) -		ipu_di_sync_config(di, cfg, 0, ARRAY_SIZE(cfg)); -	else -		ipu_di_sync_config(di, cfg_vga, 0, ARRAY_SIZE(cfg_vga)); -} - -int ipu_di_init_sync_panel(struct ipu_di *di, struct ipu_di_signal_cfg *sig) -{ -	u32 reg; -	u32 di_gen, vsync_cnt; -	u32 div; -	u32 h_total, v_total; -	int ret; -	unsigned long round; -	struct clk *parent; - -	dev_dbg(di->ipu->dev, "disp %d: panel size = %d x %d\n", -		di->id, sig->width, sig->height); - -	if ((sig->v_sync_width == 0) || (sig->h_sync_width == 0)) -		return -EINVAL; - -	if (sig->clkflags & IPU_DI_CLKMODE_EXT) -		parent = di->clk_di; -	else -		parent = di->clk_ipu; - -	ret = clk_set_parent(di->clk_di_pixel, parent); -	if (ret) { -		dev_err(di->ipu->dev, -			"setting pixel clock to parent %s failed with %d\n", -				__clk_get_name(parent), ret); -		return ret; -	} - -	if (sig->clkflags & IPU_DI_CLKMODE_SYNC) -		round = clk_get_rate(parent); -	else -		round = clk_round_rate(di->clk_di_pixel, sig->pixelclock); - -	ret = clk_set_rate(di->clk_di_pixel, round); - -	h_total = sig->width + sig->h_sync_width + sig->h_start_width + -		sig->h_end_width; -	v_total = sig->height + sig->v_sync_width + sig->v_start_width + -		sig->v_end_width; - -	mutex_lock(&di_mutex); - -	div = ipu_di_read(di, DI_BS_CLKGEN0) & 0xfff; -	div = div / 16;		/* Now divider is integer portion */ - -	/* Setup pixel clock timing */ -	/* Down time is half of period */ -	ipu_di_write(di, (div << 16), DI_BS_CLKGEN1); - -	ipu_di_data_wave_config(di, SYNC_WAVE, div - 1, div - 1); -	ipu_di_data_pin_config(di, SYNC_WAVE, DI_PIN15, 3, 0, div * 2); - -	di_gen = ipu_di_read(di, DI_GENERAL) & DI_GEN_DI_CLK_EXT; -	di_gen |= DI_GEN_DI_VSYNC_EXT; - -	if (sig->interlaced) { -		ipu_di_sync_config_interlaced(di, sig); - -		/* set y_sel = 1 */ -		di_gen |= 0x10000000; -		di_gen |= DI_GEN_POLARITY_5; -		di_gen |= DI_GEN_POLARITY_8; - -		vsync_cnt = 7; - -		if (sig->Hsync_pol) -			di_gen |= DI_GEN_POLARITY_3; -		if (sig->Vsync_pol) -			di_gen |= DI_GEN_POLARITY_2; -	} else { -		ipu_di_sync_config_noninterlaced(di, sig, div); - -		vsync_cnt = 3; -		if (di->id == 1) -			/* -			 * TODO: change only for TVEv2, parallel display -			 * uses pin 2 / 3 -			 */ -			if (!(sig->hsync_pin == 2 && sig->vsync_pin == 3)) -				vsync_cnt = 6; - -		if (sig->Hsync_pol) { -			if (sig->hsync_pin == 2) -				di_gen |= DI_GEN_POLARITY_2; -			else if (sig->hsync_pin == 4) -				di_gen |= DI_GEN_POLARITY_4; -			else if (sig->hsync_pin == 7) -				di_gen |= DI_GEN_POLARITY_7; -		} -		if (sig->Vsync_pol) { -			if (sig->vsync_pin == 3) -				di_gen |= DI_GEN_POLARITY_3; -			else if (sig->vsync_pin == 6) -				di_gen |= DI_GEN_POLARITY_6; -			else if (sig->vsync_pin == 8) -				di_gen |= DI_GEN_POLARITY_8; -		} -	} - -	if (!sig->clk_pol) -		di_gen |= DI_GEN_POLARITY_DISP_CLK; - -	ipu_di_write(di, di_gen, DI_GENERAL); - -	ipu_di_write(di, (--vsync_cnt << DI_VSYNC_SEL_OFFSET) | 0x00000002, -		     DI_SYNC_AS_GEN); - -	reg = ipu_di_read(di, DI_POL); -	reg &= ~(DI_POL_DRDY_DATA_POLARITY | DI_POL_DRDY_POLARITY_15); - -	if (sig->enable_pol) -		reg |= DI_POL_DRDY_POLARITY_15; -	if (sig->data_pol) -		reg |= DI_POL_DRDY_DATA_POLARITY; - -	ipu_di_write(di, reg, DI_POL); - -	mutex_unlock(&di_mutex); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_di_init_sync_panel); - -int ipu_di_enable(struct ipu_di *di) -{ -	int ret = clk_prepare_enable(di->clk_di_pixel); -	if (ret) -		return ret; - -	ipu_module_enable(di->ipu, di->module); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_di_enable); - -int ipu_di_disable(struct ipu_di *di) -{ -	ipu_module_disable(di->ipu, di->module); - -	clk_disable_unprepare(di->clk_di_pixel); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_di_disable); - -int ipu_di_get_num(struct ipu_di *di) -{ -	return di->id; -} -EXPORT_SYMBOL_GPL(ipu_di_get_num); - -static DEFINE_MUTEX(ipu_di_lock); - -struct ipu_di *ipu_di_get(struct ipu_soc *ipu, int disp) -{ -	struct ipu_di *di; - -	if (disp > 1) -		return ERR_PTR(-EINVAL); - -	di = ipu->di_priv[disp]; - -	mutex_lock(&ipu_di_lock); - -	if (di->inuse) { -		di = ERR_PTR(-EBUSY); -		goto out; -	} - -	di->inuse = true; -out: -	mutex_unlock(&ipu_di_lock); - -	return di; -} -EXPORT_SYMBOL_GPL(ipu_di_get); - -void ipu_di_put(struct ipu_di *di) -{ -	mutex_lock(&ipu_di_lock); - -	di->inuse = false; - -	mutex_unlock(&ipu_di_lock); -} -EXPORT_SYMBOL_GPL(ipu_di_put); - -int ipu_di_init(struct ipu_soc *ipu, struct device *dev, int id, -		unsigned long base, -		u32 module, struct clk *clk_ipu) -{ -	struct ipu_di *di; -	int ret; -	const char *di_parent[2]; -	struct clk_init_data init = { -		.ops = &clk_di_ops, -		.num_parents = 2, -		.flags = 0, -	}; - -	if (id > 1) -		return -ENODEV; - -	di = devm_kzalloc(dev, sizeof(*di), GFP_KERNEL); -	if (!di) -		return -ENOMEM; - -	ipu->di_priv[id] = di; - -	di->clk_di = devm_clk_get(dev, id ? "di1" : "di0"); -	if (IS_ERR(di->clk_di)) -		return PTR_ERR(di->clk_di); - -	di->module = module; -	di->id = id; -	di->clk_ipu = clk_ipu; -	di->base = devm_ioremap(dev, base, PAGE_SIZE); -	if (!di->base) -		return -ENOMEM; - -	di_parent[0] = __clk_get_name(di->clk_ipu); -	di_parent[1] = __clk_get_name(di->clk_di); - -	ipu_di_write(di, 0x10, DI_BS_CLKGEN0); - -	init.parent_names = (const char **)&di_parent; -	di->clk_name = kasprintf(GFP_KERNEL, "%s_di%d_pixel", -			dev_name(dev), id); -	if (!di->clk_name) -		return -ENOMEM; - -	init.name = di->clk_name; - -	di->clk_hw_out.init = &init; -	di->clk_di_pixel = clk_register(dev, &di->clk_hw_out); - -	if (IS_ERR(di->clk_di_pixel)) { -		ret = PTR_ERR(di->clk_di_pixel); -		goto failed_clk_register; -	} - -	dev_dbg(dev, "DI%d base: 0x%08lx remapped to %p\n", -			id, base, di->base); -	di->inuse = false; -	di->ipu = ipu; - -	return 0; - -failed_clk_register: - -	kfree(di->clk_name); - -	return ret; -} - -void ipu_di_exit(struct ipu_soc *ipu, int id) -{ -	struct ipu_di *di = ipu->di_priv[id]; - -	clk_unregister(di->clk_di_pixel); -	kfree(di->clk_name); -} diff --git a/drivers/staging/imx-drm/ipu-v3/ipu-dmfc.c b/drivers/staging/imx-drm/ipu-v3/ipu-dmfc.c deleted file mode 100644 index 2e97c33b81e..00000000000 --- a/drivers/staging/imx-drm/ipu-v3/ipu-dmfc.c +++ /dev/null @@ -1,418 +0,0 @@ -/* - * Copyright (c) 2010 Sascha Hauer <s.hauer@pengutronix.de> - * Copyright (C) 2005-2009 Freescale Semiconductor, Inc. - * - * This program is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License as published by the - * Free Software Foundation; either version 2 of the License, or (at your - * option) any later version. - * - * This program is distributed in the hope that it will be useful, but - * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY - * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License - * for more details. - */ -#include <linux/export.h> -#include <linux/types.h> -#include <linux/errno.h> -#include <linux/io.h> - -#include "imx-ipu-v3.h" -#include "ipu-prv.h" - -#define DMFC_RD_CHAN		0x0000 -#define DMFC_WR_CHAN		0x0004 -#define DMFC_WR_CHAN_DEF	0x0008 -#define DMFC_DP_CHAN		0x000c -#define DMFC_DP_CHAN_DEF	0x0010 -#define DMFC_GENERAL1		0x0014 -#define DMFC_GENERAL2		0x0018 -#define DMFC_IC_CTRL		0x001c -#define DMFC_STAT		0x0020 - -#define DMFC_WR_CHAN_1_28		0 -#define DMFC_WR_CHAN_2_41		8 -#define DMFC_WR_CHAN_1C_42		16 -#define DMFC_WR_CHAN_2C_43		24 - -#define DMFC_DP_CHAN_5B_23		0 -#define DMFC_DP_CHAN_5F_27		8 -#define DMFC_DP_CHAN_6B_24		16 -#define DMFC_DP_CHAN_6F_29		24 - -#define DMFC_FIFO_SIZE_64		(3 << 3) -#define DMFC_FIFO_SIZE_128		(2 << 3) -#define DMFC_FIFO_SIZE_256		(1 << 3) -#define DMFC_FIFO_SIZE_512		(0 << 3) - -#define DMFC_SEGMENT(x)			((x & 0x7) << 0) -#define DMFC_BURSTSIZE_128		(0 << 6) -#define DMFC_BURSTSIZE_64		(1 << 6) -#define DMFC_BURSTSIZE_32		(2 << 6) -#define DMFC_BURSTSIZE_16		(3 << 6) - -struct dmfc_channel_data { -	int		ipu_channel; -	unsigned long	channel_reg; -	unsigned long	shift; -	unsigned	eot_shift; -	unsigned	max_fifo_lines; -}; - -static const struct dmfc_channel_data dmfcdata[] = { -	{ -		.ipu_channel	= IPUV3_CHANNEL_MEM_BG_SYNC, -		.channel_reg	= DMFC_DP_CHAN, -		.shift		= DMFC_DP_CHAN_5B_23, -		.eot_shift	= 20, -		.max_fifo_lines	= 3, -	}, { -		.ipu_channel	= 24, -		.channel_reg	= DMFC_DP_CHAN, -		.shift		= DMFC_DP_CHAN_6B_24, -		.eot_shift	= 22, -		.max_fifo_lines	= 1, -	}, { -		.ipu_channel	= IPUV3_CHANNEL_MEM_FG_SYNC, -		.channel_reg	= DMFC_DP_CHAN, -		.shift		= DMFC_DP_CHAN_5F_27, -		.eot_shift	= 21, -		.max_fifo_lines	= 2, -	}, { -		.ipu_channel	= IPUV3_CHANNEL_MEM_DC_SYNC, -		.channel_reg	= DMFC_WR_CHAN, -		.shift		= DMFC_WR_CHAN_1_28, -		.eot_shift	= 16, -		.max_fifo_lines	= 2, -	}, { -		.ipu_channel	= 29, -		.channel_reg	= DMFC_DP_CHAN, -		.shift		= DMFC_DP_CHAN_6F_29, -		.eot_shift	= 23, -		.max_fifo_lines	= 1, -	}, -}; - -#define DMFC_NUM_CHANNELS	ARRAY_SIZE(dmfcdata) - -struct ipu_dmfc_priv; - -struct dmfc_channel { -	unsigned			slots; -	unsigned			slotmask; -	unsigned			segment; -	int				burstsize; -	struct ipu_soc			*ipu; -	struct ipu_dmfc_priv		*priv; -	const struct dmfc_channel_data	*data; -}; - -struct ipu_dmfc_priv { -	struct ipu_soc *ipu; -	struct device *dev; -	struct dmfc_channel channels[DMFC_NUM_CHANNELS]; -	struct mutex mutex; -	unsigned long bandwidth_per_slot; -	void __iomem *base; -	int use_count; -}; - -int ipu_dmfc_enable_channel(struct dmfc_channel *dmfc) -{ -	struct ipu_dmfc_priv *priv = dmfc->priv; -	mutex_lock(&priv->mutex); - -	if (!priv->use_count) -		ipu_module_enable(priv->ipu, IPU_CONF_DMFC_EN); - -	priv->use_count++; - -	mutex_unlock(&priv->mutex); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_dmfc_enable_channel); - -void ipu_dmfc_disable_channel(struct dmfc_channel *dmfc) -{ -	struct ipu_dmfc_priv *priv = dmfc->priv; - -	mutex_lock(&priv->mutex); - -	priv->use_count--; - -	if (!priv->use_count) -		ipu_module_disable(priv->ipu, IPU_CONF_DMFC_EN); - -	if (priv->use_count < 0) -		priv->use_count = 0; - -	mutex_unlock(&priv->mutex); -} -EXPORT_SYMBOL_GPL(ipu_dmfc_disable_channel); - -static int ipu_dmfc_setup_channel(struct dmfc_channel *dmfc, int slots, -		int segment, int burstsize) -{ -	struct ipu_dmfc_priv *priv = dmfc->priv; -	u32 val, field; - -	dev_dbg(priv->dev, -			"dmfc: using %d slots starting from segment %d for IPU channel %d\n", -			slots, segment, dmfc->data->ipu_channel); - -	if (!dmfc) -		return -EINVAL; - -	switch (slots) { -	case 1: -		field = DMFC_FIFO_SIZE_64; -		break; -	case 2: -		field = DMFC_FIFO_SIZE_128; -		break; -	case 4: -		field = DMFC_FIFO_SIZE_256; -		break; -	case 8: -		field = DMFC_FIFO_SIZE_512; -		break; -	default: -		return -EINVAL; -	} - -	switch (burstsize) { -	case 16: -		field |= DMFC_BURSTSIZE_16; -		break; -	case 32: -		field |= DMFC_BURSTSIZE_32; -		break; -	case 64: -		field |= DMFC_BURSTSIZE_64; -		break; -	case 128: -		field |= DMFC_BURSTSIZE_128; -		break; -	} - -	field |= DMFC_SEGMENT(segment); - -	val = readl(priv->base + dmfc->data->channel_reg); - -	val &= ~(0xff << dmfc->data->shift); -	val |= field << dmfc->data->shift; - -	writel(val, priv->base + dmfc->data->channel_reg); - -	dmfc->slots = slots; -	dmfc->segment = segment; -	dmfc->burstsize = burstsize; -	dmfc->slotmask = ((1 << slots) - 1) << segment; - -	return 0; -} - -static int dmfc_bandwidth_to_slots(struct ipu_dmfc_priv *priv, -		unsigned long bandwidth) -{ -	int slots = 1; - -	while (slots * priv->bandwidth_per_slot < bandwidth) -		slots *= 2; - -	return slots; -} - -static int dmfc_find_slots(struct ipu_dmfc_priv *priv, int slots) -{ -	unsigned slotmask_need, slotmask_used = 0; -	int i, segment = 0; - -	slotmask_need = (1 << slots) - 1; - -	for (i = 0; i < DMFC_NUM_CHANNELS; i++) -		slotmask_used |= priv->channels[i].slotmask; - -	while (slotmask_need <= 0xff) { -		if (!(slotmask_used & slotmask_need)) -			return segment; - -		slotmask_need <<= 1; -		segment++; -	} - -	return -EBUSY; -} - -void ipu_dmfc_free_bandwidth(struct dmfc_channel *dmfc) -{ -	struct ipu_dmfc_priv *priv = dmfc->priv; -	int i; - -	dev_dbg(priv->dev, "dmfc: freeing %d slots starting from segment %d\n", -			dmfc->slots, dmfc->segment); - -	mutex_lock(&priv->mutex); - -	if (!dmfc->slots) -		goto out; - -	dmfc->slotmask = 0; -	dmfc->slots = 0; -	dmfc->segment = 0; - -	for (i = 0; i < DMFC_NUM_CHANNELS; i++) -		priv->channels[i].slotmask = 0; - -	for (i = 0; i < DMFC_NUM_CHANNELS; i++) { -		if (priv->channels[i].slots > 0) { -			priv->channels[i].segment = -				dmfc_find_slots(priv, priv->channels[i].slots); -			priv->channels[i].slotmask = -				((1 << priv->channels[i].slots) - 1) << -				priv->channels[i].segment; -		} -	} - -	for (i = 0; i < DMFC_NUM_CHANNELS; i++) { -		if (priv->channels[i].slots > 0) -			ipu_dmfc_setup_channel(&priv->channels[i], -					priv->channels[i].slots, -					priv->channels[i].segment, -					priv->channels[i].burstsize); -	} -out: -	mutex_unlock(&priv->mutex); -} -EXPORT_SYMBOL_GPL(ipu_dmfc_free_bandwidth); - -int ipu_dmfc_alloc_bandwidth(struct dmfc_channel *dmfc, -		unsigned long bandwidth_pixel_per_second, int burstsize) -{ -	struct ipu_dmfc_priv *priv = dmfc->priv; -	int slots = dmfc_bandwidth_to_slots(priv, bandwidth_pixel_per_second); -	int segment = -1, ret = 0; - -	dev_dbg(priv->dev, "dmfc: trying to allocate %ldMpixel/s for IPU channel %d\n", -			bandwidth_pixel_per_second / 1000000, -			dmfc->data->ipu_channel); - -	ipu_dmfc_free_bandwidth(dmfc); - -	mutex_lock(&priv->mutex); - -	if (slots > 8) { -		ret = -EBUSY; -		goto out; -	} - -	/* Always allocate at least 128*4 bytes (2 slots) */ -	if (slots < 2) -		slots = 2; - -	/* For the MEM_BG channel, first try to allocate twice the slots */ -	if (dmfc->data->ipu_channel == IPUV3_CHANNEL_MEM_BG_SYNC) -		segment = dmfc_find_slots(priv, slots * 2); -	if (segment >= 0) -		slots *= 2; -	else -		segment = dmfc_find_slots(priv, slots); -	if (segment < 0) { -		ret = -EBUSY; -		goto out; -	} - -	ipu_dmfc_setup_channel(dmfc, slots, segment, burstsize); - -out: -	mutex_unlock(&priv->mutex); - -	return ret; -} -EXPORT_SYMBOL_GPL(ipu_dmfc_alloc_bandwidth); - -int ipu_dmfc_init_channel(struct dmfc_channel *dmfc, int width) -{ -	struct ipu_dmfc_priv *priv = dmfc->priv; -	u32 dmfc_gen1; - -	dmfc_gen1 = readl(priv->base + DMFC_GENERAL1); - -	if ((dmfc->slots * 64 * 4) / width > dmfc->data->max_fifo_lines) -		dmfc_gen1 |= 1 << dmfc->data->eot_shift; -	else -		dmfc_gen1 &= ~(1 << dmfc->data->eot_shift); - -	writel(dmfc_gen1, priv->base + DMFC_GENERAL1); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_dmfc_init_channel); - -struct dmfc_channel *ipu_dmfc_get(struct ipu_soc *ipu, int ipu_channel) -{ -	struct ipu_dmfc_priv *priv = ipu->dmfc_priv; -	int i; - -	for (i = 0; i < DMFC_NUM_CHANNELS; i++) -		if (dmfcdata[i].ipu_channel == ipu_channel) -			return &priv->channels[i]; -	return ERR_PTR(-ENODEV); -} -EXPORT_SYMBOL_GPL(ipu_dmfc_get); - -void ipu_dmfc_put(struct dmfc_channel *dmfc) -{ -	ipu_dmfc_free_bandwidth(dmfc); -} -EXPORT_SYMBOL_GPL(ipu_dmfc_put); - -int ipu_dmfc_init(struct ipu_soc *ipu, struct device *dev, unsigned long base, -		struct clk *ipu_clk) -{ -	struct ipu_dmfc_priv *priv; -	int i; - -	priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); -	if (!priv) -		return -ENOMEM; - -	priv->base = devm_ioremap(dev, base, PAGE_SIZE); -	if (!priv->base) -		return -ENOMEM; - -	priv->dev = dev; -	priv->ipu = ipu; -	mutex_init(&priv->mutex); - -	ipu->dmfc_priv = priv; - -	for (i = 0; i < DMFC_NUM_CHANNELS; i++) { -		priv->channels[i].priv = priv; -		priv->channels[i].ipu = ipu; -		priv->channels[i].data = &dmfcdata[i]; -	} - -	writel(0x0, priv->base + DMFC_WR_CHAN); -	writel(0x0, priv->base + DMFC_DP_CHAN); - -	/* -	 * We have a total bandwidth of clkrate * 4pixel divided -	 * into 8 slots. -	 */ -	priv->bandwidth_per_slot = clk_get_rate(ipu_clk) * 4 / 8; - -	dev_dbg(dev, "dmfc: 8 slots with %ldMpixel/s bandwidth each\n", -			priv->bandwidth_per_slot / 1000000); - -	writel(0x202020f6, priv->base + DMFC_WR_CHAN_DEF); -	writel(0x2020f6f6, priv->base + DMFC_DP_CHAN_DEF); -	writel(0x00000003, priv->base + DMFC_GENERAL1); - -	return 0; -} - -void ipu_dmfc_exit(struct ipu_soc *ipu) -{ -} diff --git a/drivers/staging/imx-drm/ipu-v3/ipu-dp.c b/drivers/staging/imx-drm/ipu-v3/ipu-dp.c deleted file mode 100644 index 231afd6c60f..00000000000 --- a/drivers/staging/imx-drm/ipu-v3/ipu-dp.c +++ /dev/null @@ -1,338 +0,0 @@ -/* - * Copyright (c) 2010 Sascha Hauer <s.hauer@pengutronix.de> - * Copyright (C) 2005-2009 Freescale Semiconductor, Inc. - * - * This program is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License as published by the - * Free Software Foundation; either version 2 of the License, or (at your - * option) any later version. - * - * This program is distributed in the hope that it will be useful, but - * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY - * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License - * for more details. - */ -#include <linux/export.h> -#include <linux/kernel.h> -#include <linux/types.h> -#include <linux/errno.h> -#include <linux/io.h> -#include <linux/err.h> - -#include "imx-ipu-v3.h" -#include "ipu-prv.h" - -#define DP_SYNC 0 -#define DP_ASYNC0 0x60 -#define DP_ASYNC1 0xBC - -#define DP_COM_CONF		0x0 -#define DP_GRAPH_WIND_CTRL	0x0004 -#define DP_FG_POS		0x0008 -#define DP_CSC_A_0		0x0044 -#define DP_CSC_A_1		0x0048 -#define DP_CSC_A_2		0x004C -#define DP_CSC_A_3		0x0050 -#define DP_CSC_0		0x0054 -#define DP_CSC_1		0x0058 - -#define DP_COM_CONF_FG_EN		(1 << 0) -#define DP_COM_CONF_GWSEL		(1 << 1) -#define DP_COM_CONF_GWAM		(1 << 2) -#define DP_COM_CONF_GWCKE		(1 << 3) -#define DP_COM_CONF_CSC_DEF_MASK	(3 << 8) -#define DP_COM_CONF_CSC_DEF_OFFSET	8 -#define DP_COM_CONF_CSC_DEF_FG		(3 << 8) -#define DP_COM_CONF_CSC_DEF_BG		(2 << 8) -#define DP_COM_CONF_CSC_DEF_BOTH	(1 << 8) - -#define IPUV3_NUM_FLOWS		3 - -struct ipu_dp_priv; - -struct ipu_dp { -	u32 flow; -	bool in_use; -	bool foreground; -	enum ipu_color_space in_cs; -}; - -struct ipu_flow { -	struct ipu_dp foreground; -	struct ipu_dp background; -	enum ipu_color_space out_cs; -	void __iomem *base; -	struct ipu_dp_priv *priv; -}; - -struct ipu_dp_priv { -	struct ipu_soc *ipu; -	struct device *dev; -	void __iomem *base; -	struct ipu_flow flow[IPUV3_NUM_FLOWS]; -	struct mutex mutex; -	int use_count; -}; - -static u32 ipu_dp_flow_base[] = {DP_SYNC, DP_ASYNC0, DP_ASYNC1}; - -static inline struct ipu_flow *to_flow(struct ipu_dp *dp) -{ -	if (dp->foreground) -		return container_of(dp, struct ipu_flow, foreground); -	else -		return container_of(dp, struct ipu_flow, background); -} - -int ipu_dp_set_global_alpha(struct ipu_dp *dp, bool enable, -		u8 alpha, bool bg_chan) -{ -	struct ipu_flow *flow = to_flow(dp); -	struct ipu_dp_priv *priv = flow->priv; -	u32 reg; - -	mutex_lock(&priv->mutex); - -	reg = readl(flow->base + DP_COM_CONF); -	if (bg_chan) -		reg &= ~DP_COM_CONF_GWSEL; -	else -		reg |= DP_COM_CONF_GWSEL; -	writel(reg, flow->base + DP_COM_CONF); - -	if (enable) { -		reg = readl(flow->base + DP_GRAPH_WIND_CTRL) & 0x00FFFFFFL; -		writel(reg | ((u32) alpha << 24), -			     flow->base + DP_GRAPH_WIND_CTRL); - -		reg = readl(flow->base + DP_COM_CONF); -		writel(reg | DP_COM_CONF_GWAM, flow->base + DP_COM_CONF); -	} else { -		reg = readl(flow->base + DP_COM_CONF); -		writel(reg & ~DP_COM_CONF_GWAM, flow->base + DP_COM_CONF); -	} - -	ipu_srm_dp_sync_update(priv->ipu); - -	mutex_unlock(&priv->mutex); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_dp_set_global_alpha); - -int ipu_dp_set_window_pos(struct ipu_dp *dp, u16 x_pos, u16 y_pos) -{ -	struct ipu_flow *flow = to_flow(dp); -	struct ipu_dp_priv *priv = flow->priv; - -	writel((x_pos << 16) | y_pos, flow->base + DP_FG_POS); - -	ipu_srm_dp_sync_update(priv->ipu); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_dp_set_window_pos); - -static void ipu_dp_csc_init(struct ipu_flow *flow, -		enum ipu_color_space in, -		enum ipu_color_space out, -		u32 place) -{ -	u32 reg; - -	reg = readl(flow->base + DP_COM_CONF); -	reg &= ~DP_COM_CONF_CSC_DEF_MASK; - -	if (in == out) { -		writel(reg, flow->base + DP_COM_CONF); -		return; -	} - -	if (in == IPUV3_COLORSPACE_RGB && out == IPUV3_COLORSPACE_YUV) { -		writel(0x099 | (0x12d << 16), flow->base + DP_CSC_A_0); -		writel(0x03a | (0x3a9 << 16), flow->base + DP_CSC_A_1); -		writel(0x356 | (0x100 << 16), flow->base + DP_CSC_A_2); -		writel(0x100 | (0x329 << 16), flow->base + DP_CSC_A_3); -		writel(0x3d6 | (0x0000 << 16) | (2 << 30), -				flow->base + DP_CSC_0); -		writel(0x200 | (2 << 14) | (0x200 << 16) | (2 << 30), -				flow->base + DP_CSC_1); -	} else { -		writel(0x095 | (0x000 << 16), flow->base + DP_CSC_A_0); -		writel(0x0cc | (0x095 << 16), flow->base + DP_CSC_A_1); -		writel(0x3ce | (0x398 << 16), flow->base + DP_CSC_A_2); -		writel(0x095 | (0x0ff << 16), flow->base + DP_CSC_A_3); -		writel(0x000 | (0x3e42 << 16) | (1 << 30), -				flow->base + DP_CSC_0); -		writel(0x10a | (1 << 14) | (0x3dd6 << 16) | (1 << 30), -				flow->base + DP_CSC_1); -	} - -	reg |= place; - -	writel(reg, flow->base + DP_COM_CONF); -} - -int ipu_dp_setup_channel(struct ipu_dp *dp, -		enum ipu_color_space in, -		enum ipu_color_space out) -{ -	struct ipu_flow *flow = to_flow(dp); -	struct ipu_dp_priv *priv = flow->priv; - -	mutex_lock(&priv->mutex); - -	dp->in_cs = in; - -	if (!dp->foreground) -		flow->out_cs = out; - -	if (flow->foreground.in_cs == flow->background.in_cs) { -		/* -		 * foreground and background are of same colorspace, put -		 * colorspace converter after combining unit. -		 */ -		ipu_dp_csc_init(flow, flow->foreground.in_cs, flow->out_cs, -				DP_COM_CONF_CSC_DEF_BOTH); -	} else { -		if (flow->foreground.in_cs == flow->out_cs) -			/* -			 * foreground identical to output, apply color -			 * conversion on background -			 */ -			ipu_dp_csc_init(flow, flow->background.in_cs, -					flow->out_cs, DP_COM_CONF_CSC_DEF_BG); -		else -			ipu_dp_csc_init(flow, flow->foreground.in_cs, -					flow->out_cs, DP_COM_CONF_CSC_DEF_FG); -	} - -	ipu_srm_dp_sync_update(priv->ipu); - -	mutex_unlock(&priv->mutex); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_dp_setup_channel); - -int ipu_dp_enable_channel(struct ipu_dp *dp) -{ -	struct ipu_flow *flow = to_flow(dp); -	struct ipu_dp_priv *priv = flow->priv; - -	mutex_lock(&priv->mutex); - -	if (!priv->use_count) -		ipu_module_enable(priv->ipu, IPU_CONF_DP_EN); - -	priv->use_count++; - -	if (dp->foreground) { -		u32 reg; - -		reg = readl(flow->base + DP_COM_CONF); -		reg |= DP_COM_CONF_FG_EN; -		writel(reg, flow->base + DP_COM_CONF); - -		ipu_srm_dp_sync_update(priv->ipu); -	} - -	mutex_unlock(&priv->mutex); - -	return 0; -} -EXPORT_SYMBOL_GPL(ipu_dp_enable_channel); - -void ipu_dp_disable_channel(struct ipu_dp *dp) -{ -	struct ipu_flow *flow = to_flow(dp); -	struct ipu_dp_priv *priv = flow->priv; - -	mutex_lock(&priv->mutex); - -	priv->use_count--; - -	if (dp->foreground) { -		u32 reg, csc; - -		reg = readl(flow->base + DP_COM_CONF); -		csc = reg & DP_COM_CONF_CSC_DEF_MASK; -		if (csc == DP_COM_CONF_CSC_DEF_FG) -			reg &= ~DP_COM_CONF_CSC_DEF_MASK; - -		reg &= ~DP_COM_CONF_FG_EN; -		writel(reg, flow->base + DP_COM_CONF); - -		writel(0, flow->base + DP_FG_POS); -		ipu_srm_dp_sync_update(priv->ipu); -	} - -	if (!priv->use_count) -		ipu_module_disable(priv->ipu, IPU_CONF_DP_EN); - -	if (priv->use_count < 0) -		priv->use_count = 0; - -	mutex_unlock(&priv->mutex); -} -EXPORT_SYMBOL_GPL(ipu_dp_disable_channel); - -struct ipu_dp *ipu_dp_get(struct ipu_soc *ipu, unsigned int flow) -{ -	struct ipu_dp_priv *priv = ipu->dp_priv; -	struct ipu_dp *dp; - -	if ((flow >> 1) >= IPUV3_NUM_FLOWS) -		return ERR_PTR(-EINVAL); - -	if (flow & 1) -		dp = &priv->flow[flow >> 1].foreground; -	else -		dp = &priv->flow[flow >> 1].background; - -	if (dp->in_use) -		return ERR_PTR(-EBUSY); - -	dp->in_use = true; - -	return dp; -} -EXPORT_SYMBOL_GPL(ipu_dp_get); - -void ipu_dp_put(struct ipu_dp *dp) -{ -	dp->in_use = false; -} -EXPORT_SYMBOL_GPL(ipu_dp_put); - -int ipu_dp_init(struct ipu_soc *ipu, struct device *dev, unsigned long base) -{ -	struct ipu_dp_priv *priv; -	int i; - -	priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); -	if (!priv) -		return -ENOMEM; -	priv->dev = dev; -	priv->ipu = ipu; - -	ipu->dp_priv = priv; - -	priv->base = devm_ioremap(dev, base, PAGE_SIZE); -	if (!priv->base) -		return -ENOMEM; - -	mutex_init(&priv->mutex); - -	for (i = 0; i < IPUV3_NUM_FLOWS; i++) { -		priv->flow[i].foreground.foreground = 1; -		priv->flow[i].base = priv->base + ipu_dp_flow_base[i]; -		priv->flow[i].priv = priv; -	} - -	return 0; -} - -void ipu_dp_exit(struct ipu_soc *ipu) -{ -} diff --git a/drivers/staging/imx-drm/ipu-v3/ipu-prv.h b/drivers/staging/imx-drm/ipu-v3/ipu-prv.h deleted file mode 100644 index 4df00501adc..00000000000 --- a/drivers/staging/imx-drm/ipu-v3/ipu-prv.h +++ /dev/null @@ -1,206 +0,0 @@ -/* - * Copyright (c) 2010 Sascha Hauer <s.hauer@pengutronix.de> - * Copyright (C) 2005-2009 Freescale Semiconductor, Inc. - * - * This program is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License as published by the - * Free Software Foundation; either version 2 of the License, or (at your - * option) any later version. - * - * This program is distributed in the hope that it will be useful, but - * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY - * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License - * for more details. - */ -#ifndef __IPU_PRV_H__ -#define __IPU_PRV_H__ - -struct ipu_soc; - -#include <linux/types.h> -#include <linux/device.h> -#include <linux/clk.h> -#include <linux/platform_device.h> - -#include "imx-ipu-v3.h" - -#define IPUV3_CHANNEL_CSI0			 0 -#define IPUV3_CHANNEL_CSI1			 1 -#define IPUV3_CHANNEL_CSI2			 2 -#define IPUV3_CHANNEL_CSI3			 3 -#define IPUV3_CHANNEL_MEM_BG_SYNC		23 -#define IPUV3_CHANNEL_MEM_FG_SYNC		27 -#define IPUV3_CHANNEL_MEM_DC_SYNC		28 -#define IPUV3_CHANNEL_MEM_FG_SYNC_ALPHA		31 -#define IPUV3_CHANNEL_MEM_DC_ASYNC		41 -#define IPUV3_CHANNEL_ROT_ENC_MEM		45 -#define IPUV3_CHANNEL_ROT_VF_MEM		46 -#define IPUV3_CHANNEL_ROT_PP_MEM		47 -#define IPUV3_CHANNEL_ROT_ENC_MEM_OUT		48 -#define IPUV3_CHANNEL_ROT_VF_MEM_OUT		49 -#define IPUV3_CHANNEL_ROT_PP_MEM_OUT		50 -#define IPUV3_CHANNEL_MEM_BG_SYNC_ALPHA		51 - -#define IPU_MCU_T_DEFAULT	8 -#define IPU_CM_IDMAC_REG_OFS	0x00008000 -#define IPU_CM_IC_REG_OFS	0x00020000 -#define IPU_CM_IRT_REG_OFS	0x00028000 -#define IPU_CM_CSI0_REG_OFS	0x00030000 -#define IPU_CM_CSI1_REG_OFS	0x00038000 -#define IPU_CM_SMFC_REG_OFS	0x00050000 -#define IPU_CM_DC_REG_OFS	0x00058000 -#define IPU_CM_DMFC_REG_OFS	0x00060000 - -/* Register addresses */ -/* IPU Common registers */ -#define IPU_CM_REG(offset)	(offset) - -#define IPU_CONF			IPU_CM_REG(0) - -#define IPU_SRM_PRI1			IPU_CM_REG(0x00a0) -#define IPU_SRM_PRI2			IPU_CM_REG(0x00a4) -#define IPU_FS_PROC_FLOW1		IPU_CM_REG(0x00a8) -#define IPU_FS_PROC_FLOW2		IPU_CM_REG(0x00ac) -#define IPU_FS_PROC_FLOW3		IPU_CM_REG(0x00b0) -#define IPU_FS_DISP_FLOW1		IPU_CM_REG(0x00b4) -#define IPU_FS_DISP_FLOW2		IPU_CM_REG(0x00b8) -#define IPU_SKIP			IPU_CM_REG(0x00bc) -#define IPU_DISP_ALT_CONF		IPU_CM_REG(0x00c0) -#define IPU_DISP_GEN			IPU_CM_REG(0x00c4) -#define IPU_DISP_ALT1			IPU_CM_REG(0x00c8) -#define IPU_DISP_ALT2			IPU_CM_REG(0x00cc) -#define IPU_DISP_ALT3			IPU_CM_REG(0x00d0) -#define IPU_DISP_ALT4			IPU_CM_REG(0x00d4) -#define IPU_SNOOP			IPU_CM_REG(0x00d8) -#define IPU_MEM_RST			IPU_CM_REG(0x00dc) -#define IPU_PM				IPU_CM_REG(0x00e0) -#define IPU_GPR				IPU_CM_REG(0x00e4) -#define IPU_CHA_DB_MODE_SEL(ch)		IPU_CM_REG(0x0150 + 4 * ((ch) / 32)) -#define IPU_ALT_CHA_DB_MODE_SEL(ch)	IPU_CM_REG(0x0168 + 4 * ((ch) / 32)) -#define IPU_CHA_CUR_BUF(ch)		IPU_CM_REG(0x023C + 4 * ((ch) / 32)) -#define IPU_ALT_CUR_BUF0		IPU_CM_REG(0x0244) -#define IPU_ALT_CUR_BUF1		IPU_CM_REG(0x0248) -#define IPU_SRM_STAT			IPU_CM_REG(0x024C) -#define IPU_PROC_TASK_STAT		IPU_CM_REG(0x0250) -#define IPU_DISP_TASK_STAT		IPU_CM_REG(0x0254) -#define IPU_CHA_BUF0_RDY(ch)		IPU_CM_REG(0x0268 + 4 * ((ch) / 32)) -#define IPU_CHA_BUF1_RDY(ch)		IPU_CM_REG(0x0270 + 4 * ((ch) / 32)) -#define IPU_ALT_CHA_BUF0_RDY(ch)	IPU_CM_REG(0x0278 + 4 * ((ch) / 32)) -#define IPU_ALT_CHA_BUF1_RDY(ch)	IPU_CM_REG(0x0280 + 4 * ((ch) / 32)) - -#define IPU_INT_CTRL(n)		IPU_CM_REG(0x003C + 4 * (n)) -#define IPU_INT_STAT(n)		IPU_CM_REG(0x0200 + 4 * (n)) - -#define IPU_DI0_COUNTER_RELEASE			(1 << 24) -#define IPU_DI1_COUNTER_RELEASE			(1 << 25) - -#define IPU_IDMAC_REG(offset)	(offset) - -#define IDMAC_CONF			IPU_IDMAC_REG(0x0000) -#define IDMAC_CHA_EN(ch)		IPU_IDMAC_REG(0x0004 + 4 * ((ch) / 32)) -#define IDMAC_SEP_ALPHA			IPU_IDMAC_REG(0x000c) -#define IDMAC_ALT_SEP_ALPHA		IPU_IDMAC_REG(0x0010) -#define IDMAC_CHA_PRI(ch)		IPU_IDMAC_REG(0x0014 + 4 * ((ch) / 32)) -#define IDMAC_WM_EN(ch)			IPU_IDMAC_REG(0x001c + 4 * ((ch) / 32)) -#define IDMAC_CH_LOCK_EN_1		IPU_IDMAC_REG(0x0024) -#define IDMAC_CH_LOCK_EN_2		IPU_IDMAC_REG(0x0028) -#define IDMAC_SUB_ADDR_0		IPU_IDMAC_REG(0x002c) -#define IDMAC_SUB_ADDR_1		IPU_IDMAC_REG(0x0030) -#define IDMAC_SUB_ADDR_2		IPU_IDMAC_REG(0x0034) -#define IDMAC_BAND_EN(ch)		IPU_IDMAC_REG(0x0040 + 4 * ((ch) / 32)) -#define IDMAC_CHA_BUSY(ch)		IPU_IDMAC_REG(0x0100 + 4 * ((ch) / 32)) - -#define IPU_NUM_IRQS	(32 * 15) - -enum ipu_modules { -	IPU_CONF_CSI0_EN		= (1 << 0), -	IPU_CONF_CSI1_EN		= (1 << 1), -	IPU_CONF_IC_EN			= (1 << 2), -	IPU_CONF_ROT_EN			= (1 << 3), -	IPU_CONF_ISP_EN			= (1 << 4), -	IPU_CONF_DP_EN			= (1 << 5), -	IPU_CONF_DI0_EN			= (1 << 6), -	IPU_CONF_DI1_EN			= (1 << 7), -	IPU_CONF_SMFC_EN		= (1 << 8), -	IPU_CONF_DC_EN			= (1 << 9), -	IPU_CONF_DMFC_EN		= (1 << 10), - -	IPU_CONF_VDI_EN			= (1 << 12), - -	IPU_CONF_IDMAC_DIS		= (1 << 22), - -	IPU_CONF_IC_DMFC_SEL		= (1 << 25), -	IPU_CONF_IC_DMFC_SYNC		= (1 << 26), -	IPU_CONF_VDI_DMFC_SYNC		= (1 << 27), - -	IPU_CONF_CSI0_DATA_SOURCE	= (1 << 28), -	IPU_CONF_CSI1_DATA_SOURCE	= (1 << 29), -	IPU_CONF_IC_INPUT		= (1 << 30), -	IPU_CONF_CSI_SEL		= (1 << 31), -}; - -struct ipuv3_channel { -	unsigned int num; - -	bool enabled; -	bool busy; - -	struct ipu_soc *ipu; -}; - -struct ipu_dc_priv; -struct ipu_dmfc_priv; -struct ipu_di; -struct ipu_devtype; - -struct ipu_soc { -	struct device		*dev; -	const struct ipu_devtype	*devtype; -	enum ipuv3_type		ipu_type; -	spinlock_t		lock; -	struct mutex		channel_lock; - -	void __iomem		*cm_reg; -	void __iomem		*idmac_reg; -	struct ipu_ch_param __iomem	*cpmem_base; - -	int			usecount; - -	struct clk		*clk; - -	struct ipuv3_channel	channel[64]; - -	int			irq_sync; -	int			irq_err; -	struct irq_domain	*domain; - -	struct ipu_dc_priv	*dc_priv; -	struct ipu_dp_priv	*dp_priv; -	struct ipu_dmfc_priv	*dmfc_priv; -	struct ipu_di		*di_priv[2]; -}; - -void ipu_srm_dp_sync_update(struct ipu_soc *ipu); - -int ipu_module_enable(struct ipu_soc *ipu, u32 mask); -int ipu_module_disable(struct ipu_soc *ipu, u32 mask); - -int ipu_di_init(struct ipu_soc *ipu, struct device *dev, int id, -		unsigned long base, u32 module, struct clk *ipu_clk); -void ipu_di_exit(struct ipu_soc *ipu, int id); - -int ipu_dmfc_init(struct ipu_soc *ipu, struct device *dev, unsigned long base, -		struct clk *ipu_clk); -void ipu_dmfc_exit(struct ipu_soc *ipu); - -int ipu_dp_init(struct ipu_soc *ipu, struct device *dev, unsigned long base); -void ipu_dp_exit(struct ipu_soc *ipu); - -int ipu_dc_init(struct ipu_soc *ipu, struct device *dev, unsigned long base, -		unsigned long template_base); -void ipu_dc_exit(struct ipu_soc *ipu); - -int ipu_cpmem_init(struct ipu_soc *ipu, struct device *dev, unsigned long base); -void ipu_cpmem_exit(struct ipu_soc *ipu); - -#endif				/* __IPU_PRV_H__ */ diff --git a/drivers/staging/imx-drm/ipuv3-crtc.c b/drivers/staging/imx-drm/ipuv3-crtc.c index 6fd37a7453e..720868bff35 100644 --- a/drivers/staging/imx-drm/ipuv3-crtc.c +++ b/drivers/staging/imx-drm/ipuv3-crtc.c @@ -17,6 +17,7 @@   * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,   * MA 02110-1301, USA.   */ +#include <linux/component.h>  #include <linux/module.h>  #include <linux/export.h>  #include <linux/device.h> @@ -25,29 +26,25 @@  #include <drm/drm_crtc_helper.h>  #include <linux/fb.h>  #include <linux/clk.h> +#include <linux/errno.h>  #include <drm/drm_gem_cma_helper.h>  #include <drm/drm_fb_cma_helper.h> -#include "ipu-v3/imx-ipu-v3.h" +#include <video/imx-ipu-v3.h>  #include "imx-drm.h" +#include "ipuv3-plane.h"  #define DRIVER_DESC		"i.MX IPUv3 Graphics" -struct ipu_framebuffer { -	struct drm_framebuffer	base; -	void			*virt; -	dma_addr_t		phys; -	size_t			len; -}; -  struct ipu_crtc {  	struct device		*dev;  	struct drm_crtc		base;  	struct imx_drm_crtc	*imx_crtc; -	struct ipuv3_channel	*ipu_ch; + +	/* plane[0] is the full plane, plane[1] is the partial plane */ +	struct ipu_plane	*plane[2]; +  	struct ipu_dc		*dc; -	struct ipu_dp		*dp; -	struct dmfc_channel	*dmfc;  	struct ipu_di		*di;  	int			enabled;  	struct drm_pending_vblank_event *page_flip_event; @@ -61,50 +58,34 @@ struct ipu_crtc {  #define to_ipu_crtc(x) container_of(x, struct ipu_crtc, base) -static int calc_vref(struct drm_display_mode *mode) -{ -	unsigned long htotal, vtotal; - -	htotal = mode->htotal; -	vtotal = mode->vtotal; - -	if (!htotal || !vtotal) -		return 60; - -	return mode->clock * 1000 / vtotal / htotal; -} - -static int calc_bandwidth(struct drm_display_mode *mode, unsigned int vref) -{ -	return mode->hdisplay * mode->vdisplay * vref; -} -  static void ipu_fb_enable(struct ipu_crtc *ipu_crtc)  { +	struct ipu_soc *ipu = dev_get_drvdata(ipu_crtc->dev->parent); +  	if (ipu_crtc->enabled)  		return; -	ipu_di_enable(ipu_crtc->di); -	ipu_dmfc_enable_channel(ipu_crtc->dmfc); -	ipu_idmac_enable_channel(ipu_crtc->ipu_ch); +	ipu_dc_enable(ipu); +	ipu_plane_enable(ipu_crtc->plane[0]); +	/* Start DC channel and DI after IDMAC */  	ipu_dc_enable_channel(ipu_crtc->dc); -	if (ipu_crtc->dp) -		ipu_dp_enable_channel(ipu_crtc->dp); +	ipu_di_enable(ipu_crtc->di);  	ipu_crtc->enabled = 1;  }  static void ipu_fb_disable(struct ipu_crtc *ipu_crtc)  { +	struct ipu_soc *ipu = dev_get_drvdata(ipu_crtc->dev->parent); +  	if (!ipu_crtc->enabled)  		return; -	if (ipu_crtc->dp) -		ipu_dp_disable_channel(ipu_crtc->dp); +	/* Stop DC channel and DI before IDMAC */  	ipu_dc_disable_channel(ipu_crtc->dc); -	ipu_idmac_disable_channel(ipu_crtc->ipu_ch); -	ipu_dmfc_disable_channel(ipu_crtc->dmfc);  	ipu_di_disable(ipu_crtc->di); +	ipu_plane_disable(ipu_crtc->plane[0]); +	ipu_dc_disable(ipu);  	ipu_crtc->enabled = 0;  } @@ -148,7 +129,7 @@ static int ipu_page_flip(struct drm_crtc *crtc,  	ipu_crtc->newfb = fb;  	ipu_crtc->page_flip_event = event; -	crtc->fb = fb; +	crtc->primary->fb = fb;  	return 0;  } @@ -159,33 +140,6 @@ static const struct drm_crtc_funcs ipu_crtc_funcs = {  	.page_flip = ipu_page_flip,  }; -static int ipu_drm_set_base(struct drm_crtc *crtc, int x, int y) -{ -	struct ipu_crtc *ipu_crtc = to_ipu_crtc(crtc); -	struct drm_gem_cma_object *cma_obj; -	struct drm_framebuffer *fb = crtc->fb; -	unsigned long phys; - -	cma_obj = drm_fb_cma_get_gem_obj(fb, 0); -	if (!cma_obj) { -		DRM_LOG_KMS("entry is null.\n"); -		return -EFAULT; -	} - -	phys = cma_obj->paddr; -	phys += x * (fb->bits_per_pixel >> 3); -	phys += y * fb->pitches[0]; - -	dev_dbg(ipu_crtc->dev, "%s: phys: 0x%lx\n", __func__, phys); -	dev_dbg(ipu_crtc->dev, "%s: xy: %dx%d\n", __func__, x, y); - -	ipu_cpmem_set_stride(ipu_get_cpmem(ipu_crtc->ipu_ch), fb->pitches[0]); -	ipu_cpmem_set_buffer(ipu_get_cpmem(ipu_crtc->ipu_ch), -			  0, phys); - -	return 0; -} -  static int ipu_crtc_mode_set(struct drm_crtc *crtc,  			       struct drm_display_mode *orig_mode,  			       struct drm_display_mode *mode, @@ -193,41 +147,15 @@ static int ipu_crtc_mode_set(struct drm_crtc *crtc,  			       struct drm_framebuffer *old_fb)  {  	struct ipu_crtc *ipu_crtc = to_ipu_crtc(crtc); -	struct drm_framebuffer *fb = ipu_crtc->base.fb;  	int ret;  	struct ipu_di_signal_cfg sig_cfg = {};  	u32 out_pixel_fmt; -	struct ipu_ch_param __iomem *cpmem = ipu_get_cpmem(ipu_crtc->ipu_ch); -	int bpp; -	u32 v4l2_fmt;  	dev_dbg(ipu_crtc->dev, "%s: mode->hdisplay: %d\n", __func__,  			mode->hdisplay);  	dev_dbg(ipu_crtc->dev, "%s: mode->vdisplay: %d\n", __func__,  			mode->vdisplay); -	ipu_ch_param_zero(cpmem); - -	switch (fb->pixel_format) { -	case DRM_FORMAT_XRGB8888: -	case DRM_FORMAT_ARGB8888: -		v4l2_fmt = V4L2_PIX_FMT_RGB32; -		bpp = 32; -		break; -	case DRM_FORMAT_RGB565: -		v4l2_fmt = V4L2_PIX_FMT_RGB565; -		bpp = 16; -		break; -	case DRM_FORMAT_RGB888: -		v4l2_fmt = V4L2_PIX_FMT_RGB24; -		bpp = 24; -		break; -	default: -		dev_err(ipu_crtc->dev, "unsupported pixel format 0x%08x\n", -				fb->pixel_format); -		return -EINVAL; -	} -  	out_pixel_fmt = ipu_crtc->interface_pix_fmt;  	if (mode->flags & DRM_MODE_FLAG_INTERLACE) @@ -257,18 +185,6 @@ static int ipu_crtc_mode_set(struct drm_crtc *crtc,  	sig_cfg.hsync_pin = ipu_crtc->di_hsync_pin;  	sig_cfg.vsync_pin = ipu_crtc->di_vsync_pin; -	if (ipu_crtc->dp) { -		ret = ipu_dp_setup_channel(ipu_crtc->dp, IPUV3_COLORSPACE_RGB, -				IPUV3_COLORSPACE_RGB); -		if (ret) { -			dev_err(ipu_crtc->dev, -				"initializing display processor failed with %d\n", -				ret); -			return ret; -		} -		ipu_dp_set_global_alpha(ipu_crtc->dp, 1, 0, 1); -	} -  	ret = ipu_dc_init_sync(ipu_crtc->dc, ipu_crtc->di, sig_cfg.interlaced,  			out_pixel_fmt, mode->hdisplay);  	if (ret) { @@ -285,30 +201,9 @@ static int ipu_crtc_mode_set(struct drm_crtc *crtc,  		return ret;  	} -	ipu_cpmem_set_resolution(cpmem, mode->hdisplay, mode->vdisplay); -	ipu_cpmem_set_fmt(cpmem, v4l2_fmt); -	ipu_cpmem_set_high_priority(ipu_crtc->ipu_ch); - -	ret = ipu_dmfc_init_channel(ipu_crtc->dmfc, mode->hdisplay); -	if (ret) { -		dev_err(ipu_crtc->dev, -				"initializing dmfc channel failed with %d\n", -				ret); -		return ret; -	} - -	ret = ipu_dmfc_alloc_bandwidth(ipu_crtc->dmfc, -			calc_bandwidth(mode, calc_vref(mode)), 64); -	if (ret) { -		dev_err(ipu_crtc->dev, -				"allocating dmfc bandwidth failed with %d\n", -				ret); -		return ret; -	} - -	ipu_drm_set_base(crtc, x, y); - -	return 0; +	return ipu_plane_mode_set(ipu_crtc->plane[0], crtc, mode, crtc->primary->fb, +				  0, 0, mode->hdisplay, mode->vdisplay, +				  x, y, mode->hdisplay, mode->vdisplay);  }  static void ipu_crtc_handle_pageflip(struct ipu_crtc *ipu_crtc) @@ -332,7 +227,8 @@ static irqreturn_t ipu_irq_handler(int irq, void *dev_id)  	if (ipu_crtc->newfb) {  		ipu_crtc->newfb = NULL; -		ipu_drm_set_base(&ipu_crtc->base, 0, 0); +		ipu_plane_set_base(ipu_crtc->plane[0], ipu_crtc->base.primary->fb, +				ipu_crtc->plane[0]->x, ipu_crtc->plane[0]->y);  		ipu_crtc_handle_pageflip(ipu_crtc);  	} @@ -370,10 +266,6 @@ static struct drm_crtc_helper_funcs ipu_helper_funcs = {  static int ipu_enable_vblank(struct drm_crtc *crtc)  { -	struct ipu_crtc *ipu_crtc = to_ipu_crtc(crtc); - -	enable_irq(ipu_crtc->irq); -  	return 0;  } @@ -381,7 +273,8 @@ static void ipu_disable_vblank(struct drm_crtc *crtc)  {  	struct ipu_crtc *ipu_crtc = to_ipu_crtc(crtc); -	disable_irq(ipu_crtc->irq); +	ipu_crtc->page_flip_event = NULL; +	ipu_crtc->newfb = NULL;  }  static int ipu_set_interface_pix_fmt(struct drm_crtc *crtc, u32 encoder_type, @@ -400,6 +293,7 @@ static int ipu_set_interface_pix_fmt(struct drm_crtc *crtc, u32 encoder_type,  		ipu_crtc->di_clkflags = IPU_DI_CLKMODE_SYNC |  			IPU_DI_CLKMODE_EXT;  		break; +	case DRM_MODE_ENCODER_TMDS:  	case DRM_MODE_ENCODER_NONE:  		ipu_crtc->di_clkflags = 0;  		break; @@ -418,12 +312,8 @@ static const struct imx_drm_crtc_helper_funcs ipu_crtc_helper_funcs = {  static void ipu_put_resources(struct ipu_crtc *ipu_crtc)  { -	if (!IS_ERR_OR_NULL(ipu_crtc->ipu_ch)) -		ipu_idmac_put(ipu_crtc->ipu_ch); -	if (!IS_ERR_OR_NULL(ipu_crtc->dmfc)) -		ipu_dmfc_put(ipu_crtc->dmfc); -	if (!IS_ERR_OR_NULL(ipu_crtc->dp)) -		ipu_dp_put(ipu_crtc->dp); +	if (!IS_ERR_OR_NULL(ipu_crtc->dc)) +		ipu_dc_put(ipu_crtc->dc);  	if (!IS_ERR_OR_NULL(ipu_crtc->di))  		ipu_di_put(ipu_crtc->di);  } @@ -434,32 +324,12 @@ static int ipu_get_resources(struct ipu_crtc *ipu_crtc,  	struct ipu_soc *ipu = dev_get_drvdata(ipu_crtc->dev->parent);  	int ret; -	ipu_crtc->ipu_ch = ipu_idmac_get(ipu, pdata->dma[0]); -	if (IS_ERR(ipu_crtc->ipu_ch)) { -		ret = PTR_ERR(ipu_crtc->ipu_ch); -		goto err_out; -	} -  	ipu_crtc->dc = ipu_dc_get(ipu, pdata->dc);  	if (IS_ERR(ipu_crtc->dc)) {  		ret = PTR_ERR(ipu_crtc->dc);  		goto err_out;  	} -	ipu_crtc->dmfc = ipu_dmfc_get(ipu, pdata->dma[0]); -	if (IS_ERR(ipu_crtc->dmfc)) { -		ret = PTR_ERR(ipu_crtc->dmfc); -		goto err_out; -	} - -	if (pdata->dp >= 0) { -		ipu_crtc->dp = ipu_dp_get(ipu, pdata->dp); -		if (IS_ERR(ipu_crtc->dp)) { -			ret = PTR_ERR(ipu_crtc->dp); -			goto err_out; -		} -	} -  	ipu_crtc->di = ipu_di_get(ipu, pdata->di);  	if (IS_ERR(ipu_crtc->di)) {  		ret = PTR_ERR(ipu_crtc->di); @@ -474,10 +344,12 @@ err_out:  }  static int ipu_crtc_init(struct ipu_crtc *ipu_crtc, -		struct ipu_client_platformdata *pdata) +	struct ipu_client_platformdata *pdata, struct drm_device *drm)  {  	struct ipu_soc *ipu = dev_get_drvdata(ipu_crtc->dev->parent); +	int dp = -EINVAL;  	int ret; +	int id;  	ret = ipu_get_resources(ipu_crtc, pdata);  	if (ret) { @@ -486,68 +358,145 @@ static int ipu_crtc_init(struct ipu_crtc *ipu_crtc,  		return ret;  	} -	ret = imx_drm_add_crtc(&ipu_crtc->base, -			&ipu_crtc->imx_crtc, -			&ipu_crtc_helper_funcs, THIS_MODULE, -			ipu_crtc->dev->parent->of_node, pdata->di); +	ret = imx_drm_add_crtc(drm, &ipu_crtc->base, &ipu_crtc->imx_crtc, +			&ipu_crtc_helper_funcs, ipu_crtc->dev->of_node);  	if (ret) {  		dev_err(ipu_crtc->dev, "adding crtc failed with %d.\n", ret);  		goto err_put_resources;  	} -	ipu_crtc->irq = ipu_idmac_channel_irq(ipu, ipu_crtc->ipu_ch, -			IPU_IRQ_EOF); +	if (pdata->dp >= 0) +		dp = IPU_DP_FLOW_SYNC_BG; +	id = imx_drm_crtc_id(ipu_crtc->imx_crtc); +	ipu_crtc->plane[0] = ipu_plane_init(ipu_crtc->base.dev, ipu, +					    pdata->dma[0], dp, BIT(id), true); +	ret = ipu_plane_get_resources(ipu_crtc->plane[0]); +	if (ret) { +		dev_err(ipu_crtc->dev, "getting plane 0 resources failed with %d.\n", +			ret); +		goto err_remove_crtc; +	} + +	/* If this crtc is using the DP, add an overlay plane */ +	if (pdata->dp >= 0 && pdata->dma[1] > 0) { +		ipu_crtc->plane[1] = ipu_plane_init(ipu_crtc->base.dev, ipu, +						    pdata->dma[1], +						    IPU_DP_FLOW_SYNC_FG, +						    BIT(id), false); +		if (IS_ERR(ipu_crtc->plane[1])) +			ipu_crtc->plane[1] = NULL; +	} + +	ipu_crtc->irq = ipu_plane_irq(ipu_crtc->plane[0]);  	ret = devm_request_irq(ipu_crtc->dev, ipu_crtc->irq, ipu_irq_handler, 0,  			"imx_drm", ipu_crtc);  	if (ret < 0) {  		dev_err(ipu_crtc->dev, "irq request failed with %d.\n", ret); -		goto err_put_resources; +		goto err_put_plane_res;  	} -	disable_irq(ipu_crtc->irq); -  	return 0; +err_put_plane_res: +	ipu_plane_put_resources(ipu_crtc->plane[0]); +err_remove_crtc: +	imx_drm_remove_crtc(ipu_crtc->imx_crtc);  err_put_resources:  	ipu_put_resources(ipu_crtc);  	return ret;  } -static int ipu_drm_probe(struct platform_device *pdev) +static struct device_node *ipu_drm_get_port_by_id(struct device_node *parent, +						  int port_id)  { -	struct ipu_client_platformdata *pdata = pdev->dev.platform_data; -	struct ipu_crtc *ipu_crtc; -	int ret; +	struct device_node *port; +	int id, ret; + +	port = of_get_child_by_name(parent, "port"); +	while (port) { +		ret = of_property_read_u32(port, "reg", &id); +		if (!ret && id == port_id) +			return port; + +		do { +			port = of_get_next_child(parent, port); +			if (!port) +				return NULL; +		} while (of_node_cmp(port->name, "port")); +	} -	if (!pdata) -		return -EINVAL; +	return NULL; +} -	pdev->dev.coherent_dma_mask = DMA_BIT_MASK(32); +static int ipu_drm_bind(struct device *dev, struct device *master, void *data) +{ +	struct ipu_client_platformdata *pdata = dev->platform_data; +	struct drm_device *drm = data; +	struct ipu_crtc *ipu_crtc; +	int ret; -	ipu_crtc = devm_kzalloc(&pdev->dev, sizeof(*ipu_crtc), GFP_KERNEL); +	ipu_crtc = devm_kzalloc(dev, sizeof(*ipu_crtc), GFP_KERNEL);  	if (!ipu_crtc)  		return -ENOMEM; -	ipu_crtc->dev = &pdev->dev; +	ipu_crtc->dev = dev; -	ret = ipu_crtc_init(ipu_crtc, pdata); +	ret = ipu_crtc_init(ipu_crtc, pdata, drm);  	if (ret)  		return ret; -	platform_set_drvdata(pdev, ipu_crtc); +	dev_set_drvdata(dev, ipu_crtc);  	return 0;  } -static int ipu_drm_remove(struct platform_device *pdev) +static void ipu_drm_unbind(struct device *dev, struct device *master, +	void *data)  { -	struct ipu_crtc *ipu_crtc = platform_get_drvdata(pdev); +	struct ipu_crtc *ipu_crtc = dev_get_drvdata(dev);  	imx_drm_remove_crtc(ipu_crtc->imx_crtc); +	ipu_plane_put_resources(ipu_crtc->plane[0]);  	ipu_put_resources(ipu_crtc); +} + +static const struct component_ops ipu_crtc_ops = { +	.bind = ipu_drm_bind, +	.unbind = ipu_drm_unbind, +}; + +static int ipu_drm_probe(struct platform_device *pdev) +{ +	struct device *dev = &pdev->dev; +	struct ipu_client_platformdata *pdata = dev->platform_data; +	int ret; + +	if (!dev->platform_data) +		return -EINVAL; +	if (!dev->of_node) { +		/* Associate crtc device with the corresponding DI port node */ +		dev->of_node = ipu_drm_get_port_by_id(dev->parent->of_node, +						      pdata->di + 2); +		if (!dev->of_node) { +			dev_err(dev, "missing port@%d node in %s\n", +				pdata->di + 2, dev->parent->of_node->full_name); +			return -ENODEV; +		} +	} + +	ret = dma_set_coherent_mask(dev, DMA_BIT_MASK(32)); +	if (ret) +		return ret; + +	return component_add(dev, &ipu_crtc_ops); +} + +static int ipu_drm_remove(struct platform_device *pdev) +{ +	component_del(&pdev->dev, &ipu_crtc_ops);  	return 0;  } diff --git a/drivers/staging/imx-drm/ipuv3-plane.c b/drivers/staging/imx-drm/ipuv3-plane.c new file mode 100644 index 00000000000..6f393a11f44 --- /dev/null +++ b/drivers/staging/imx-drm/ipuv3-plane.c @@ -0,0 +1,387 @@ +/* + * i.MX IPUv3 DP Overlay Planes + * + * Copyright (C) 2013 Philipp Zabel, Pengutronix + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the + * GNU General Public License for more details. + */ + +#include <drm/drmP.h> +#include <drm/drm_fb_cma_helper.h> +#include <drm/drm_gem_cma_helper.h> + +#include "video/imx-ipu-v3.h" +#include "ipuv3-plane.h" + +#define to_ipu_plane(x)	container_of(x, struct ipu_plane, base) + +static const uint32_t ipu_plane_formats[] = { +	DRM_FORMAT_XRGB1555, +	DRM_FORMAT_XBGR1555, +	DRM_FORMAT_ARGB8888, +	DRM_FORMAT_XRGB8888, +	DRM_FORMAT_ABGR8888, +	DRM_FORMAT_XBGR8888, +	DRM_FORMAT_YUYV, +	DRM_FORMAT_YVYU, +	DRM_FORMAT_YUV420, +	DRM_FORMAT_YVU420, +}; + +int ipu_plane_irq(struct ipu_plane *ipu_plane) +{ +	return ipu_idmac_channel_irq(ipu_plane->ipu, ipu_plane->ipu_ch, +				     IPU_IRQ_EOF); +} + +static int calc_vref(struct drm_display_mode *mode) +{ +	unsigned long htotal, vtotal; + +	htotal = mode->htotal; +	vtotal = mode->vtotal; + +	if (!htotal || !vtotal) +		return 60; + +	return DIV_ROUND_UP(mode->clock * 1000, vtotal * htotal); +} + +static inline int calc_bandwidth(int width, int height, unsigned int vref) +{ +	return width * height * vref; +} + +int ipu_plane_set_base(struct ipu_plane *ipu_plane, struct drm_framebuffer *fb, +		       int x, int y) +{ +	struct ipu_ch_param __iomem *cpmem; +	struct drm_gem_cma_object *cma_obj; +	unsigned long eba; + +	cma_obj = drm_fb_cma_get_gem_obj(fb, 0); +	if (!cma_obj) { +		DRM_DEBUG_KMS("entry is null.\n"); +		return -EFAULT; +	} + +	dev_dbg(ipu_plane->base.dev->dev, "phys = %pad, x = %d, y = %d", +		&cma_obj->paddr, x, y); + +	cpmem = ipu_get_cpmem(ipu_plane->ipu_ch); +	ipu_cpmem_set_stride(cpmem, fb->pitches[0]); + +	eba = cma_obj->paddr + fb->offsets[0] + +	      fb->pitches[0] * y + (fb->bits_per_pixel >> 3) * x; +	ipu_cpmem_set_buffer(cpmem, 0, eba); +	ipu_cpmem_set_buffer(cpmem, 1, eba); + +	/* cache offsets for subsequent pageflips */ +	ipu_plane->x = x; +	ipu_plane->y = y; + +	return 0; +} + +int ipu_plane_mode_set(struct ipu_plane *ipu_plane, struct drm_crtc *crtc, +		       struct drm_display_mode *mode, +		       struct drm_framebuffer *fb, int crtc_x, int crtc_y, +		       unsigned int crtc_w, unsigned int crtc_h, +		       uint32_t src_x, uint32_t src_y, +		       uint32_t src_w, uint32_t src_h) +{ +	struct ipu_ch_param __iomem *cpmem; +	struct device *dev = ipu_plane->base.dev->dev; +	int ret; + +	/* no scaling */ +	if (src_w != crtc_w || src_h != crtc_h) +		return -EINVAL; + +	/* clip to crtc bounds */ +	if (crtc_x < 0) { +		if (-crtc_x > crtc_w) +			return -EINVAL; +		src_x += -crtc_x; +		src_w -= -crtc_x; +		crtc_w -= -crtc_x; +		crtc_x = 0; +	} +	if (crtc_y < 0) { +		if (-crtc_y > crtc_h) +			return -EINVAL; +		src_y += -crtc_y; +		src_h -= -crtc_y; +		crtc_h -= -crtc_y; +		crtc_y = 0; +	} +	if (crtc_x + crtc_w > mode->hdisplay) { +		if (crtc_x > mode->hdisplay) +			return -EINVAL; +		crtc_w = mode->hdisplay - crtc_x; +		src_w = crtc_w; +	} +	if (crtc_y + crtc_h > mode->vdisplay) { +		if (crtc_y > mode->vdisplay) +			return -EINVAL; +		crtc_h = mode->vdisplay - crtc_y; +		src_h = crtc_h; +	} +	/* full plane minimum width is 13 pixels */ +	if (crtc_w < 13 && (ipu_plane->dp_flow != IPU_DP_FLOW_SYNC_FG)) +		return -EINVAL; +	if (crtc_h < 2) +		return -EINVAL; + +	switch (ipu_plane->dp_flow) { +	case IPU_DP_FLOW_SYNC_BG: +		ret = ipu_dp_setup_channel(ipu_plane->dp, +				IPUV3_COLORSPACE_RGB, +				IPUV3_COLORSPACE_RGB); +		if (ret) { +			dev_err(dev, +				"initializing display processor failed with %d\n", +				ret); +			return ret; +		} +		ipu_dp_set_global_alpha(ipu_plane->dp, 1, 0, 1); +		break; +	case IPU_DP_FLOW_SYNC_FG: +		ipu_dp_setup_channel(ipu_plane->dp, +				ipu_drm_fourcc_to_colorspace(fb->pixel_format), +				IPUV3_COLORSPACE_UNKNOWN); +		ipu_dp_set_window_pos(ipu_plane->dp, crtc_x, crtc_y); +		break; +	} + +	ret = ipu_dmfc_init_channel(ipu_plane->dmfc, crtc_w); +	if (ret) { +		dev_err(dev, "initializing dmfc channel failed with %d\n", ret); +		return ret; +	} + +	ret = ipu_dmfc_alloc_bandwidth(ipu_plane->dmfc, +			calc_bandwidth(crtc_w, crtc_h, +				       calc_vref(mode)), 64); +	if (ret) { +		dev_err(dev, "allocating dmfc bandwidth failed with %d\n", ret); +		return ret; +	} + +	cpmem = ipu_get_cpmem(ipu_plane->ipu_ch); +	ipu_ch_param_zero(cpmem); +	ipu_cpmem_set_resolution(cpmem, src_w, src_h); +	ret = ipu_cpmem_set_fmt(cpmem, fb->pixel_format); +	if (ret < 0) { +		dev_err(dev, "unsupported pixel format 0x%08x\n", +			fb->pixel_format); +		return ret; +	} +	ipu_cpmem_set_high_priority(ipu_plane->ipu_ch); + +	ret = ipu_plane_set_base(ipu_plane, fb, src_x, src_y); +	if (ret < 0) +		return ret; + +	return 0; +} + +void ipu_plane_put_resources(struct ipu_plane *ipu_plane) +{ +	if (!IS_ERR_OR_NULL(ipu_plane->dp)) +		ipu_dp_put(ipu_plane->dp); +	if (!IS_ERR_OR_NULL(ipu_plane->dmfc)) +		ipu_dmfc_put(ipu_plane->dmfc); +	if (!IS_ERR_OR_NULL(ipu_plane->ipu_ch)) +		ipu_idmac_put(ipu_plane->ipu_ch); +} + +int ipu_plane_get_resources(struct ipu_plane *ipu_plane) +{ +	int ret; + +	ipu_plane->ipu_ch = ipu_idmac_get(ipu_plane->ipu, ipu_plane->dma); +	if (IS_ERR(ipu_plane->ipu_ch)) { +		ret = PTR_ERR(ipu_plane->ipu_ch); +		DRM_ERROR("failed to get idmac channel: %d\n", ret); +		return ret; +	} + +	ipu_plane->dmfc = ipu_dmfc_get(ipu_plane->ipu, ipu_plane->dma); +	if (IS_ERR(ipu_plane->dmfc)) { +		ret = PTR_ERR(ipu_plane->dmfc); +		DRM_ERROR("failed to get dmfc: ret %d\n", ret); +		goto err_out; +	} + +	if (ipu_plane->dp_flow >= 0) { +		ipu_plane->dp = ipu_dp_get(ipu_plane->ipu, ipu_plane->dp_flow); +		if (IS_ERR(ipu_plane->dp)) { +			ret = PTR_ERR(ipu_plane->dp); +			DRM_ERROR("failed to get dp flow: %d\n", ret); +			goto err_out; +		} +	} + +	return 0; +err_out: +	ipu_plane_put_resources(ipu_plane); + +	return ret; +} + +void ipu_plane_enable(struct ipu_plane *ipu_plane) +{ +	if (ipu_plane->dp) +		ipu_dp_enable(ipu_plane->ipu); +	ipu_dmfc_enable_channel(ipu_plane->dmfc); +	ipu_idmac_enable_channel(ipu_plane->ipu_ch); +	if (ipu_plane->dp) +		ipu_dp_enable_channel(ipu_plane->dp); + +	ipu_plane->enabled = true; +} + +void ipu_plane_disable(struct ipu_plane *ipu_plane) +{ +	ipu_plane->enabled = false; + +	ipu_idmac_wait_busy(ipu_plane->ipu_ch, 50); + +	if (ipu_plane->dp) +		ipu_dp_disable_channel(ipu_plane->dp); +	ipu_idmac_disable_channel(ipu_plane->ipu_ch); +	ipu_dmfc_disable_channel(ipu_plane->dmfc); +	if (ipu_plane->dp) +		ipu_dp_disable(ipu_plane->ipu); +} + +static void ipu_plane_dpms(struct ipu_plane *ipu_plane, int mode) +{ +	bool enable; + +	DRM_DEBUG_KMS("mode = %d", mode); + +	enable = (mode == DRM_MODE_DPMS_ON); + +	if (enable == ipu_plane->enabled) +		return; + +	if (enable) { +		ipu_plane_enable(ipu_plane); +	} else { +		ipu_plane_disable(ipu_plane); + +		ipu_idmac_put(ipu_plane->ipu_ch); +		ipu_dmfc_put(ipu_plane->dmfc); +		ipu_dp_put(ipu_plane->dp); +	} +} + +/* + * drm_plane API + */ + +static int ipu_update_plane(struct drm_plane *plane, struct drm_crtc *crtc, +			    struct drm_framebuffer *fb, int crtc_x, int crtc_y, +			    unsigned int crtc_w, unsigned int crtc_h, +			    uint32_t src_x, uint32_t src_y, +			    uint32_t src_w, uint32_t src_h) +{ +	struct ipu_plane *ipu_plane = to_ipu_plane(plane); +	int ret = 0; + +	DRM_DEBUG_KMS("plane - %p\n", plane); + +	if (!ipu_plane->enabled) +		ret = ipu_plane_get_resources(ipu_plane); +	if (ret < 0) +		return ret; + +	ret = ipu_plane_mode_set(ipu_plane, crtc, &crtc->hwmode, fb, +			crtc_x, crtc_y, crtc_w, crtc_h, +			src_x >> 16, src_y >> 16, src_w >> 16, src_h >> 16); +	if (ret < 0) { +		ipu_plane_put_resources(ipu_plane); +		return ret; +	} + +	if (crtc != plane->crtc) +		dev_info(plane->dev->dev, "crtc change: %p -> %p\n", +				plane->crtc, crtc); +	plane->crtc = crtc; + +	ipu_plane_dpms(ipu_plane, DRM_MODE_DPMS_ON); + +	return 0; +} + +static int ipu_disable_plane(struct drm_plane *plane) +{ +	struct ipu_plane *ipu_plane = to_ipu_plane(plane); + +	DRM_DEBUG_KMS("[%d] %s\n", __LINE__, __func__); + +	ipu_plane_dpms(ipu_plane, DRM_MODE_DPMS_OFF); + +	ipu_plane_put_resources(ipu_plane); + +	return 0; +} + +static void ipu_plane_destroy(struct drm_plane *plane) +{ +	struct ipu_plane *ipu_plane = to_ipu_plane(plane); + +	DRM_DEBUG_KMS("[%d] %s\n", __LINE__, __func__); + +	ipu_disable_plane(plane); +	drm_plane_cleanup(plane); +	kfree(ipu_plane); +} + +static struct drm_plane_funcs ipu_plane_funcs = { +	.update_plane	= ipu_update_plane, +	.disable_plane	= ipu_disable_plane, +	.destroy	= ipu_plane_destroy, +}; + +struct ipu_plane *ipu_plane_init(struct drm_device *dev, struct ipu_soc *ipu, +				 int dma, int dp, unsigned int possible_crtcs, +				 bool priv) +{ +	struct ipu_plane *ipu_plane; +	int ret; + +	DRM_DEBUG_KMS("channel %d, dp flow %d, possible_crtcs=0x%x\n", +		      dma, dp, possible_crtcs); + +	ipu_plane = kzalloc(sizeof(*ipu_plane), GFP_KERNEL); +	if (!ipu_plane) { +		DRM_ERROR("failed to allocate plane\n"); +		return ERR_PTR(-ENOMEM); +	} + +	ipu_plane->ipu = ipu; +	ipu_plane->dma = dma; +	ipu_plane->dp_flow = dp; + +	ret = drm_plane_init(dev, &ipu_plane->base, possible_crtcs, +			     &ipu_plane_funcs, ipu_plane_formats, +			     ARRAY_SIZE(ipu_plane_formats), +			     priv); +	if (ret) { +		DRM_ERROR("failed to initialize plane\n"); +		kfree(ipu_plane); +		return ERR_PTR(ret); +	} + +	return ipu_plane; +} diff --git a/drivers/staging/imx-drm/ipuv3-plane.h b/drivers/staging/imx-drm/ipuv3-plane.h new file mode 100644 index 00000000000..c0aae5bcb5d --- /dev/null +++ b/drivers/staging/imx-drm/ipuv3-plane.h @@ -0,0 +1,55 @@ +#ifndef __IPUV3_PLANE_H__ +#define __IPUV3_PLANE_H__ + +#include <drm/drm_crtc.h> /* drm_plane */ + +struct drm_plane; +struct drm_device; +struct ipu_soc; +struct drm_crtc; +struct drm_framebuffer; + +struct ipuv3_channel; +struct dmfc_channel; +struct ipu_dp; + +struct ipu_plane { +	struct drm_plane	base; + +	struct ipu_soc		*ipu; +	struct ipuv3_channel	*ipu_ch; +	struct dmfc_channel	*dmfc; +	struct ipu_dp		*dp; + +	int			dma; +	int			dp_flow; + +	int			x; +	int			y; + +	bool			enabled; +}; + +struct ipu_plane *ipu_plane_init(struct drm_device *dev, struct ipu_soc *ipu, +				 int dma, int dp, unsigned int possible_crtcs, +				 bool priv); + +/* Init IDMAC, DMFC, DP */ +int ipu_plane_mode_set(struct ipu_plane *plane, struct drm_crtc *crtc, +		       struct drm_display_mode *mode, +		       struct drm_framebuffer *fb, int crtc_x, int crtc_y, +		       unsigned int crtc_w, unsigned int crtc_h, +		       uint32_t src_x, uint32_t src_y, uint32_t src_w, +		       uint32_t src_h); + +void ipu_plane_enable(struct ipu_plane *plane); +void ipu_plane_disable(struct ipu_plane *plane); +int ipu_plane_set_base(struct ipu_plane *plane, struct drm_framebuffer *fb, +		       int x, int y); + +int ipu_plane_get_resources(struct ipu_plane *plane); +void ipu_plane_put_resources(struct ipu_plane *plane); + +int ipu_plane_irq(struct ipu_plane *plane); + +#endif diff --git a/drivers/staging/imx-drm/parallel-display.c b/drivers/staging/imx-drm/parallel-display.c index 24aa9beedcf..4ca61afdf62 100644 --- a/drivers/staging/imx-drm/parallel-display.c +++ b/drivers/staging/imx-drm/parallel-display.c @@ -18,11 +18,14 @@   * MA 02110-1301, USA.   */ +#include <linux/component.h>  #include <linux/module.h>  #include <drm/drmP.h>  #include <drm/drm_fb_helper.h>  #include <drm/drm_crtc_helper.h> +#include <drm/drm_panel.h>  #include <linux/videodev2.h> +#include <video/of_display_timing.h>  #include "imx-drm.h" @@ -31,15 +34,14 @@  struct imx_parallel_display {  	struct drm_connector connector; -	struct imx_drm_connector *imx_drm_connector;  	struct drm_encoder encoder; -	struct imx_drm_encoder *imx_drm_encoder;  	struct device *dev;  	void *edid;  	int edid_len;  	u32 interface_pix_fmt;  	int mode_valid;  	struct drm_display_mode mode; +	struct drm_panel *panel;  };  static enum drm_connector_status imx_pd_connector_detect( @@ -48,17 +50,19 @@ static enum drm_connector_status imx_pd_connector_detect(  	return connector_status_connected;  } -static void imx_pd_connector_destroy(struct drm_connector *connector) -{ -	/* do not free here */ -} -  static int imx_pd_connector_get_modes(struct drm_connector *connector)  {  	struct imx_parallel_display *imxpd = con_to_imxpd(connector);  	struct device_node *np = imxpd->dev->of_node;  	int num_modes = 0; +	if (imxpd->panel && imxpd->panel->funcs && +	    imxpd->panel->funcs->get_modes) { +		num_modes = imxpd->panel->funcs->get_modes(imxpd->panel); +		if (num_modes > 0) +			return num_modes; +	} +  	if (imxpd->edid) {  		drm_mode_connector_update_edid_property(connector, imxpd->edid);  		num_modes = drm_add_edid_modes(connector, imxpd->edid); @@ -66,6 +70,8 @@ static int imx_pd_connector_get_modes(struct drm_connector *connector)  	if (imxpd->mode_valid) {  		struct drm_display_mode *mode = drm_mode_create(connector->dev); +		if (!mode) +			return -EINVAL;  		drm_mode_copy(mode, &imxpd->mode);  		mode->type |= DRM_MODE_TYPE_DRIVER | DRM_MODE_TYPE_PREFERRED,  		drm_mode_probed_add(connector, mode); @@ -74,7 +80,9 @@ static int imx_pd_connector_get_modes(struct drm_connector *connector)  	if (np) {  		struct drm_display_mode *mode = drm_mode_create(connector->dev); -		of_get_drm_display_mode(np, &imxpd->mode, 0); +		if (!mode) +			return -EINVAL; +		of_get_drm_display_mode(np, &imxpd->mode, OF_USE_NATIVE_MODE);  		drm_mode_copy(mode, &imxpd->mode);  		mode->type |= DRM_MODE_TYPE_DRIVER | DRM_MODE_TYPE_PREFERRED,  		drm_mode_probed_add(connector, mode); @@ -84,12 +92,6 @@ static int imx_pd_connector_get_modes(struct drm_connector *connector)  	return num_modes;  } -static int imx_pd_connector_mode_valid(struct drm_connector *connector, -			  struct drm_display_mode *mode) -{ -	return 0; -} -  static struct drm_encoder *imx_pd_connector_best_encoder(  		struct drm_connector *connector)  { @@ -100,6 +102,12 @@ static struct drm_encoder *imx_pd_connector_best_encoder(  static void imx_pd_encoder_dpms(struct drm_encoder *encoder, int mode)  { +	struct imx_parallel_display *imxpd = enc_to_imxpd(encoder); + +	if (mode != DRM_MODE_DPMS_ON) +		drm_panel_disable(imxpd->panel); +	else +		drm_panel_enable(imxpd->panel);  }  static bool imx_pd_encoder_mode_fixup(struct drm_encoder *encoder, @@ -113,8 +121,7 @@ static void imx_pd_encoder_prepare(struct drm_encoder *encoder)  {  	struct imx_parallel_display *imxpd = enc_to_imxpd(encoder); -	imx_drm_crtc_panel_format(encoder->crtc, DRM_MODE_ENCODER_NONE, -			imxpd->interface_pix_fmt); +	imx_drm_panel_format(encoder, imxpd->interface_pix_fmt);  }  static void imx_pd_encoder_commit(struct drm_encoder *encoder) @@ -131,26 +138,20 @@ static void imx_pd_encoder_disable(struct drm_encoder *encoder)  {  } -static void imx_pd_encoder_destroy(struct drm_encoder *encoder) -{ -	/* do not free here */ -} -  static struct drm_connector_funcs imx_pd_connector_funcs = {  	.dpms = drm_helper_connector_dpms,  	.fill_modes = drm_helper_probe_single_connector_modes,  	.detect = imx_pd_connector_detect, -	.destroy = imx_pd_connector_destroy, +	.destroy = imx_drm_connector_destroy,  };  static struct drm_connector_helper_funcs imx_pd_connector_helper_funcs = {  	.get_modes = imx_pd_connector_get_modes,  	.best_encoder = imx_pd_connector_best_encoder, -	.mode_valid = imx_pd_connector_mode_valid,  };  static struct drm_encoder_funcs imx_pd_encoder_funcs = { -	.destroy = imx_pd_encoder_destroy, +	.destroy = imx_drm_encoder_destroy,  };  static struct drm_encoder_helper_funcs imx_pd_encoder_helper_funcs = { @@ -162,51 +163,53 @@ static struct drm_encoder_helper_funcs imx_pd_encoder_helper_funcs = {  	.disable = imx_pd_encoder_disable,  }; -static int imx_pd_register(struct imx_parallel_display *imxpd) +static int imx_pd_register(struct drm_device *drm, +	struct imx_parallel_display *imxpd)  {  	int ret; -	drm_mode_connector_attach_encoder(&imxpd->connector, &imxpd->encoder); - -	imxpd->connector.funcs = &imx_pd_connector_funcs; -	imxpd->encoder.funcs = &imx_pd_encoder_funcs; +	ret = imx_drm_encoder_parse_of(drm, &imxpd->encoder, +				       imxpd->dev->of_node); +	if (ret) +		return ret; -	imxpd->encoder.encoder_type = DRM_MODE_ENCODER_NONE; -	imxpd->connector.connector_type = DRM_MODE_CONNECTOR_VGA; +	/* set the connector's dpms to OFF so that +	 * drm_helper_connector_dpms() won't return +	 * immediately since the current state is ON +	 * at this point. +	 */ +	imxpd->connector.dpms = DRM_MODE_DPMS_OFF;  	drm_encoder_helper_add(&imxpd->encoder, &imx_pd_encoder_helper_funcs); -	ret = imx_drm_add_encoder(&imxpd->encoder, &imxpd->imx_drm_encoder, -			THIS_MODULE); -	if (ret) { -		dev_err(imxpd->dev, "adding encoder failed with %d\n", ret); -		return ret; -	} +	drm_encoder_init(drm, &imxpd->encoder, &imx_pd_encoder_funcs, +			 DRM_MODE_ENCODER_NONE);  	drm_connector_helper_add(&imxpd->connector,  			&imx_pd_connector_helper_funcs); +	drm_connector_init(drm, &imxpd->connector, &imx_pd_connector_funcs, +			   DRM_MODE_CONNECTOR_VGA); -	ret = imx_drm_add_connector(&imxpd->connector, -			&imxpd->imx_drm_connector, THIS_MODULE); -	if (ret) { -		imx_drm_remove_encoder(imxpd->imx_drm_encoder); -		dev_err(imxpd->dev, "adding connector failed with %d\n", ret); -		return ret; -	} +	if (imxpd->panel) +		drm_panel_attach(imxpd->panel, &imxpd->connector); + +	drm_mode_connector_attach_encoder(&imxpd->connector, &imxpd->encoder);  	imxpd->connector.encoder = &imxpd->encoder;  	return 0;  } -static int imx_pd_probe(struct platform_device *pdev) +static int imx_pd_bind(struct device *dev, struct device *master, void *data)  { -	struct device_node *np = pdev->dev.of_node; +	struct drm_device *drm = data; +	struct device_node *np = dev->of_node; +	struct device_node *panel_node;  	const u8 *edidp;  	struct imx_parallel_display *imxpd;  	int ret;  	const char *fmt; -	imxpd = devm_kzalloc(&pdev->dev, sizeof(*imxpd), GFP_KERNEL); +	imxpd = devm_kzalloc(dev, sizeof(*imxpd), GFP_KERNEL);  	if (!imxpd)  		return -ENOMEM; @@ -222,32 +225,47 @@ static int imx_pd_probe(struct platform_device *pdev)  			imxpd->interface_pix_fmt = V4L2_PIX_FMT_RGB565;  		else if (!strcmp(fmt, "bgr666"))  			imxpd->interface_pix_fmt = V4L2_PIX_FMT_BGR666; +		else if (!strcmp(fmt, "lvds666")) +			imxpd->interface_pix_fmt = v4l2_fourcc('L', 'V', 'D', '6');  	} -	imxpd->dev = &pdev->dev; +	panel_node = of_parse_phandle(np, "fsl,panel", 0); +	if (panel_node) +		imxpd->panel = of_drm_find_panel(panel_node); -	ret = imx_pd_register(imxpd); +	imxpd->dev = dev; + +	ret = imx_pd_register(drm, imxpd);  	if (ret)  		return ret; -	ret = imx_drm_encoder_add_possible_crtcs(imxpd->imx_drm_encoder, np); - -	platform_set_drvdata(pdev, imxpd); +	dev_set_drvdata(dev, imxpd);  	return 0;  } -static int imx_pd_remove(struct platform_device *pdev) +static void imx_pd_unbind(struct device *dev, struct device *master, +	void *data)  { -	struct imx_parallel_display *imxpd = platform_get_drvdata(pdev); -	struct drm_connector *connector = &imxpd->connector; -	struct drm_encoder *encoder = &imxpd->encoder; +	struct imx_parallel_display *imxpd = dev_get_drvdata(dev); -	drm_mode_connector_detach_encoder(connector, encoder); +	imxpd->encoder.funcs->destroy(&imxpd->encoder); +	imxpd->connector.funcs->destroy(&imxpd->connector); +} -	imx_drm_remove_connector(imxpd->imx_drm_connector); -	imx_drm_remove_encoder(imxpd->imx_drm_encoder); +static const struct component_ops imx_pd_ops = { +	.bind	= imx_pd_bind, +	.unbind	= imx_pd_unbind, +}; +static int imx_pd_probe(struct platform_device *pdev) +{ +	return component_add(&pdev->dev, &imx_pd_ops); +} + +static int imx_pd_remove(struct platform_device *pdev) +{ +	component_del(&pdev->dev, &imx_pd_ops);  	return 0;  }  | 
