2009-11-03 12:23:50 +03:00
/*
* linux / drivers / video / omap2 / dss / core . c
*
* Copyright ( C ) 2009 Nokia Corporation
* Author : Tomi Valkeinen < tomi . valkeinen @ nokia . com >
*
* Some code and ideas taken from drivers / video / omap / driver
* by Imre Deak .
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License version 2 as published by
* the Free Software Foundation .
*
* 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 .
*
* You should have received a copy of the GNU General Public License along with
* this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# define DSS_SUBSYS_NAME "CORE"
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/clk.h>
# include <linux/err.h>
# include <linux/platform_device.h>
# include <linux/seq_file.h>
# include <linux/debugfs.h>
# include <linux/io.h>
# include <linux/device.h>
2010-02-04 18:03:41 +03:00
# include <linux/regulator/consumer.h>
OMAPDSS: Use PM notifiers for system suspend
The current way how omapdss handles system suspend and resume is that
omapdss device (a platform device, which is not part of the device
hierarchy of the DSS HW devices, like DISPC and DSI, or panels.) uses
the suspend and resume callbacks from platform_driver to handle system
suspend. It does this by disabling all enabled panels on suspend, and
resuming the previously disabled panels on resume.
This presents a few problems.
One is that as omapdss device is not related to the panel devices or the
DSS HW devices, there's no ordering in the suspend process. This means
that suspend could be first ran for DSS HW devices and panels, and only
then for omapdss device. Currently this is not a problem, as DSS HW
devices and panels do not handle suspend.
Another, more pressing problem, is that when suspending or resuming, the
runtime PM functions return -EACCES as runtime PM is disabled during
system suspend. This causes the driver to print warnings, and operations
to fail as they think that they failed to bring up the HW.
This patch changes the omapdss suspend handling to use PM notifiers,
which are called before suspend and after resume. This way we have a
normally functioning system when we are suspending and resuming the
panels.
This patch, I believe, creates a problem that somebody could enable or
disable a panel between PM_SUSPEND_PREPARE and the system suspend, and
similarly the other way around in resume. I choose to ignore the problem
for now, as it sounds rather unlikely, and if it happens, it's not
fatal.
In the long run the system suspend handling of omapdss and panels should
be thought out properly. The current approach feels rather hacky.
Perhaps the panel drivers should handle system suspend, or the users of
omapdss (omapfb, omapdrm) should handle system suspend.
Note that after this patch we could probably revert
0eaf9f52e94f756147dbfe1faf1f77a02378dbf9 (OMAPDSS: use sync versions of
pm_runtime_put). But as I said, this patch may be temporary, so let's
leave the sync version still in place.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Reported-by: Jassi Brar <jaswinder.singh@linaro.org>
Tested-by: Jassi Brar <jaswinder.singh@linaro.org>
Tested-by: Joe Woodward <jw@terrafix.co.uk>
Signed-off-by: Archit Taneja <archit@ti.com>
[fts: fixed 2 brace coding style issues]
Signed-off-by: Florian Tobias Schandinat <FlorianSchandinat@gmx.de>
2012-07-04 16:43:48 +04:00
# include <linux/suspend.h>
2012-09-10 14:58:29 +04:00
# include <linux/slab.h>
2009-11-03 12:23:50 +03:00
2011-05-11 15:05:07 +04:00
# include <video/omapdss.h>
2009-11-03 12:23:50 +03:00
# include "dss.h"
2010-09-15 17:50:00 +04:00
# include "dss_features.h"
2009-11-03 12:23:50 +03:00
static struct {
struct platform_device * pdev ;
2010-02-04 18:03:41 +03:00
2012-02-23 15:00:51 +04:00
const char * default_display_name ;
2009-11-03 12:23:50 +03:00
} core ;
static char * def_disp_name ;
module_param_named ( def_disp , def_disp_name , charp , 0 ) ;
2011-02-16 13:53:44 +03:00
MODULE_PARM_DESC ( def_disp , " default display name " ) ;
2009-11-03 12:23:50 +03:00
2013-05-23 13:07:50 +04:00
static bool dss_initialized ;
2012-10-29 14:40:46 +04:00
const char * omapdss_get_default_display_name ( void )
2012-09-06 15:26:10 +04:00
{
return core . default_display_name ;
}
2012-10-29 14:40:46 +04:00
EXPORT_SYMBOL ( omapdss_get_default_display_name ) ;
2012-09-06 15:26:10 +04:00
2012-10-18 14:46:29 +04:00
enum omapdss_version omapdss_get_version ( void )
{
struct omap_dss_board_info * pdata = core . pdev - > dev . platform_data ;
return pdata - > version ;
}
EXPORT_SYMBOL ( omapdss_get_version ) ;
2013-05-23 13:07:50 +04:00
bool omapdss_is_initialized ( void )
{
return dss_initialized ;
}
EXPORT_SYMBOL ( omapdss_is_initialized ) ;
2012-10-10 11:46:06 +04:00
struct platform_device * dss_get_core_pdev ( void )
{
return core . pdev ;
}
2012-02-20 13:50:06 +04:00
int dss_dsi_enable_pads ( int dsi_id , unsigned lane_mask )
{
struct omap_dss_board_info * board_data = core . pdev - > dev . platform_data ;
if ( ! board_data - > dsi_enable_pads )
return - ENOENT ;
return board_data - > dsi_enable_pads ( dsi_id , lane_mask ) ;
}
void dss_dsi_disable_pads ( int dsi_id , unsigned lane_mask )
{
struct omap_dss_board_info * board_data = core . pdev - > dev . platform_data ;
2012-10-16 15:51:21 +04:00
if ( ! board_data - > dsi_disable_pads )
2012-02-20 13:50:06 +04:00
return ;
return board_data - > dsi_disable_pads ( dsi_id , lane_mask ) ;
}
2012-03-08 14:52:38 +04:00
int dss_set_min_bus_tput ( struct device * dev , unsigned long tput )
{
struct omap_dss_board_info * pdata = core . pdev - > dev . platform_data ;
if ( pdata - > set_min_bus_tput )
return pdata - > set_min_bus_tput ( dev , tput ) ;
else
return 0 ;
}
2012-09-29 09:55:42 +04:00
# if defined(CONFIG_OMAP2_DSS_DEBUGFS)
2009-11-03 12:23:50 +03:00
static int dss_debug_show ( struct seq_file * s , void * unused )
{
void ( * func ) ( struct seq_file * ) = s - > private ;
func ( s ) ;
return 0 ;
}
static int dss_debug_open ( struct inode * inode , struct file * file )
{
return single_open ( file , dss_debug_show , inode - > i_private ) ;
}
static const struct file_operations dss_debug_fops = {
. open = dss_debug_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct dentry * dss_debugfs_dir ;
static int dss_initialize_debugfs ( void )
{
dss_debugfs_dir = debugfs_create_dir ( " omapdss " , NULL ) ;
if ( IS_ERR ( dss_debugfs_dir ) ) {
int err = PTR_ERR ( dss_debugfs_dir ) ;
dss_debugfs_dir = NULL ;
return err ;
}
debugfs_create_file ( " clk " , S_IRUGO , dss_debugfs_dir ,
& dss_debug_dump_clocks , & dss_debug_fops ) ;
return 0 ;
}
static void dss_uninitialize_debugfs ( void )
{
if ( dss_debugfs_dir )
debugfs_remove_recursive ( dss_debugfs_dir ) ;
}
2012-03-02 20:01:07 +04:00
int dss_debugfs_create_file ( const char * name , void ( * write ) ( struct seq_file * ) )
{
struct dentry * d ;
d = debugfs_create_file ( name , S_IRUGO , dss_debugfs_dir ,
write , & dss_debug_fops ) ;
2013-03-19 12:03:14 +04:00
return PTR_RET ( d ) ;
2012-03-02 20:01:07 +04:00
}
2012-09-29 09:55:42 +04:00
# else /* CONFIG_OMAP2_DSS_DEBUGFS */
2010-05-07 13:58:41 +04:00
static inline int dss_initialize_debugfs ( void )
{
return 0 ;
}
static inline void dss_uninitialize_debugfs ( void )
{
}
2012-05-23 17:45:09 +04:00
int dss_debugfs_create_file ( const char * name , void ( * write ) ( struct seq_file * ) )
2012-03-02 20:01:07 +04:00
{
return 0 ;
}
2012-09-29 09:55:42 +04:00
# endif /* CONFIG_OMAP2_DSS_DEBUGFS */
2009-11-03 12:23:50 +03:00
/* PLATFORM DEVICE */
OMAPDSS: Use PM notifiers for system suspend
The current way how omapdss handles system suspend and resume is that
omapdss device (a platform device, which is not part of the device
hierarchy of the DSS HW devices, like DISPC and DSI, or panels.) uses
the suspend and resume callbacks from platform_driver to handle system
suspend. It does this by disabling all enabled panels on suspend, and
resuming the previously disabled panels on resume.
This presents a few problems.
One is that as omapdss device is not related to the panel devices or the
DSS HW devices, there's no ordering in the suspend process. This means
that suspend could be first ran for DSS HW devices and panels, and only
then for omapdss device. Currently this is not a problem, as DSS HW
devices and panels do not handle suspend.
Another, more pressing problem, is that when suspending or resuming, the
runtime PM functions return -EACCES as runtime PM is disabled during
system suspend. This causes the driver to print warnings, and operations
to fail as they think that they failed to bring up the HW.
This patch changes the omapdss suspend handling to use PM notifiers,
which are called before suspend and after resume. This way we have a
normally functioning system when we are suspending and resuming the
panels.
This patch, I believe, creates a problem that somebody could enable or
disable a panel between PM_SUSPEND_PREPARE and the system suspend, and
similarly the other way around in resume. I choose to ignore the problem
for now, as it sounds rather unlikely, and if it happens, it's not
fatal.
In the long run the system suspend handling of omapdss and panels should
be thought out properly. The current approach feels rather hacky.
Perhaps the panel drivers should handle system suspend, or the users of
omapdss (omapfb, omapdrm) should handle system suspend.
Note that after this patch we could probably revert
0eaf9f52e94f756147dbfe1faf1f77a02378dbf9 (OMAPDSS: use sync versions of
pm_runtime_put). But as I said, this patch may be temporary, so let's
leave the sync version still in place.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Reported-by: Jassi Brar <jaswinder.singh@linaro.org>
Tested-by: Jassi Brar <jaswinder.singh@linaro.org>
Tested-by: Joe Woodward <jw@terrafix.co.uk>
Signed-off-by: Archit Taneja <archit@ti.com>
[fts: fixed 2 brace coding style issues]
Signed-off-by: Florian Tobias Schandinat <FlorianSchandinat@gmx.de>
2012-07-04 16:43:48 +04:00
static int omap_dss_pm_notif ( struct notifier_block * b , unsigned long v , void * d )
{
DSSDBG ( " pm notif %lu \n " , v ) ;
switch ( v ) {
case PM_SUSPEND_PREPARE :
DSSDBG ( " suspending displays \n " ) ;
return dss_suspend_all_devices ( ) ;
case PM_POST_SUSPEND :
DSSDBG ( " resuming displays \n " ) ;
return dss_resume_all_devices ( ) ;
default :
return 0 ;
}
}
static struct notifier_block omap_dss_pm_notif_block = {
. notifier_call = omap_dss_pm_notif ,
} ;
2012-02-17 19:41:13 +04:00
static int __init omap_dss_probe ( struct platform_device * pdev )
2009-11-03 12:23:50 +03:00
{
struct omap_dss_board_info * pdata = pdev - > dev . platform_data ;
int r ;
core . pdev = pdev ;
2012-10-18 14:46:29 +04:00
dss_features_init ( omapdss_get_version ( ) ) ;
2010-09-15 17:50:00 +04:00
2009-11-03 12:23:50 +03:00
r = dss_initialize_debugfs ( ) ;
if ( r )
2010-05-07 13:58:42 +04:00
goto err_debugfs ;
2009-11-03 12:23:50 +03:00
2012-02-23 15:00:51 +04:00
if ( def_disp_name )
core . default_display_name = def_disp_name ;
2012-11-16 16:59:56 +04:00
else if ( pdata - > default_display_name )
core . default_display_name = pdata - > default_display_name ;
2012-02-23 15:00:51 +04:00
else if ( pdata - > default_device )
core . default_display_name = pdata - > default_device - > name ;
OMAPDSS: Use PM notifiers for system suspend
The current way how omapdss handles system suspend and resume is that
omapdss device (a platform device, which is not part of the device
hierarchy of the DSS HW devices, like DISPC and DSI, or panels.) uses
the suspend and resume callbacks from platform_driver to handle system
suspend. It does this by disabling all enabled panels on suspend, and
resuming the previously disabled panels on resume.
This presents a few problems.
One is that as omapdss device is not related to the panel devices or the
DSS HW devices, there's no ordering in the suspend process. This means
that suspend could be first ran for DSS HW devices and panels, and only
then for omapdss device. Currently this is not a problem, as DSS HW
devices and panels do not handle suspend.
Another, more pressing problem, is that when suspending or resuming, the
runtime PM functions return -EACCES as runtime PM is disabled during
system suspend. This causes the driver to print warnings, and operations
to fail as they think that they failed to bring up the HW.
This patch changes the omapdss suspend handling to use PM notifiers,
which are called before suspend and after resume. This way we have a
normally functioning system when we are suspending and resuming the
panels.
This patch, I believe, creates a problem that somebody could enable or
disable a panel between PM_SUSPEND_PREPARE and the system suspend, and
similarly the other way around in resume. I choose to ignore the problem
for now, as it sounds rather unlikely, and if it happens, it's not
fatal.
In the long run the system suspend handling of omapdss and panels should
be thought out properly. The current approach feels rather hacky.
Perhaps the panel drivers should handle system suspend, or the users of
omapdss (omapfb, omapdrm) should handle system suspend.
Note that after this patch we could probably revert
0eaf9f52e94f756147dbfe1faf1f77a02378dbf9 (OMAPDSS: use sync versions of
pm_runtime_put). But as I said, this patch may be temporary, so let's
leave the sync version still in place.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Reported-by: Jassi Brar <jaswinder.singh@linaro.org>
Tested-by: Jassi Brar <jaswinder.singh@linaro.org>
Tested-by: Joe Woodward <jw@terrafix.co.uk>
Signed-off-by: Archit Taneja <archit@ti.com>
[fts: fixed 2 brace coding style issues]
Signed-off-by: Florian Tobias Schandinat <FlorianSchandinat@gmx.de>
2012-07-04 16:43:48 +04:00
register_pm_notifier ( & omap_dss_pm_notif_block ) ;
2009-11-03 12:23:50 +03:00
return 0 ;
2010-05-07 13:58:42 +04:00
err_debugfs :
2009-11-03 12:23:50 +03:00
return r ;
}
static int omap_dss_remove ( struct platform_device * pdev )
{
OMAPDSS: Use PM notifiers for system suspend
The current way how omapdss handles system suspend and resume is that
omapdss device (a platform device, which is not part of the device
hierarchy of the DSS HW devices, like DISPC and DSI, or panels.) uses
the suspend and resume callbacks from platform_driver to handle system
suspend. It does this by disabling all enabled panels on suspend, and
resuming the previously disabled panels on resume.
This presents a few problems.
One is that as omapdss device is not related to the panel devices or the
DSS HW devices, there's no ordering in the suspend process. This means
that suspend could be first ran for DSS HW devices and panels, and only
then for omapdss device. Currently this is not a problem, as DSS HW
devices and panels do not handle suspend.
Another, more pressing problem, is that when suspending or resuming, the
runtime PM functions return -EACCES as runtime PM is disabled during
system suspend. This causes the driver to print warnings, and operations
to fail as they think that they failed to bring up the HW.
This patch changes the omapdss suspend handling to use PM notifiers,
which are called before suspend and after resume. This way we have a
normally functioning system when we are suspending and resuming the
panels.
This patch, I believe, creates a problem that somebody could enable or
disable a panel between PM_SUSPEND_PREPARE and the system suspend, and
similarly the other way around in resume. I choose to ignore the problem
for now, as it sounds rather unlikely, and if it happens, it's not
fatal.
In the long run the system suspend handling of omapdss and panels should
be thought out properly. The current approach feels rather hacky.
Perhaps the panel drivers should handle system suspend, or the users of
omapdss (omapfb, omapdrm) should handle system suspend.
Note that after this patch we could probably revert
0eaf9f52e94f756147dbfe1faf1f77a02378dbf9 (OMAPDSS: use sync versions of
pm_runtime_put). But as I said, this patch may be temporary, so let's
leave the sync version still in place.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Reported-by: Jassi Brar <jaswinder.singh@linaro.org>
Tested-by: Jassi Brar <jaswinder.singh@linaro.org>
Tested-by: Joe Woodward <jw@terrafix.co.uk>
Signed-off-by: Archit Taneja <archit@ti.com>
[fts: fixed 2 brace coding style issues]
Signed-off-by: Florian Tobias Schandinat <FlorianSchandinat@gmx.de>
2012-07-04 16:43:48 +04:00
unregister_pm_notifier ( & omap_dss_pm_notif_block ) ;
2009-11-03 12:23:50 +03:00
dss_uninitialize_debugfs ( ) ;
return 0 ;
}
static void omap_dss_shutdown ( struct platform_device * pdev )
{
DSSDBG ( " shutdown \n " ) ;
dss_disable_all_devices ( ) ;
}
static struct platform_driver omap_dss_driver = {
. remove = omap_dss_remove ,
. shutdown = omap_dss_shutdown ,
. driver = {
. name = " omapdss " ,
. owner = THIS_MODULE ,
} ,
} ;
/* BUS */
static int dss_bus_match ( struct device * dev , struct device_driver * driver )
{
struct omap_dss_device * dssdev = to_dss_device ( dev ) ;
DSSDBG ( " bus_match. dev %s/%s, drv %s \n " ,
dev_name ( dev ) , dssdev - > driver_name , driver - > name ) ;
return strcmp ( dssdev - > driver_name , driver - > name ) = = 0 ;
}
static struct bus_type dss_bus_type = {
. name = " omapdss " ,
. match = dss_bus_match ,
} ;
static void dss_bus_release ( struct device * dev )
{
DSSDBG ( " bus_release \n " ) ;
}
static struct device dss_bus = {
. release = dss_bus_release ,
} ;
struct bus_type * dss_get_bus ( void )
{
return & dss_bus_type ;
}
/* DRIVER */
static int dss_driver_probe ( struct device * dev )
{
int r ;
struct omap_dss_driver * dssdrv = to_dss_driver ( dev - > driver ) ;
struct omap_dss_device * dssdev = to_dss_device ( dev ) ;
DSSDBG ( " driver_probe: dev %s/%s, drv %s \n " ,
dev_name ( dev ) , dssdev - > driver_name ,
dssdrv - > driver . name ) ;
r = dssdrv - > probe ( dssdev ) ;
if ( r ) {
DSSERR ( " driver probe failed: %d \n " , r ) ;
return r ;
}
DSSDBG ( " probe done for device %s \n " , dev_name ( dev ) ) ;
dssdev - > driver = dssdrv ;
return 0 ;
}
static int dss_driver_remove ( struct device * dev )
{
struct omap_dss_driver * dssdrv = to_dss_driver ( dev - > driver ) ;
struct omap_dss_device * dssdev = to_dss_device ( dev ) ;
DSSDBG ( " driver_remove: dev %s/%s \n " , dev_name ( dev ) ,
dssdev - > driver_name ) ;
dssdrv - > remove ( dssdev ) ;
dssdev - > driver = NULL ;
return 0 ;
}
OMAPDSS: Implement display (dis)connect support
We currently have two steps in panel initialization and startup: probing
and enabling. After the panel has been probed, it's ready and can be
configured and later enabled.
This model is not enough with more complex display pipelines, where we
may have, for example, two panels, of which only one can be used at a
time, connected to the same video output.
To support that kind of scenarios, we need to add new step to the
initialization: connect.
This patch adds support for connecting and disconnecting panels. After
probe, but before connect, no panel ops should be called. When the
connect is called, a proper video pipeline is established, and the panel
is ready for use. If some part in the video pipeline is already
connected (by some other panel), the connect call fails.
One key difference with the old style setup is that connect() handles
also connecting to the overlay manager. This means that the omapfb (or
omapdrm) no longer needs to figure out which overlay manager to use, but
it can just call connect() on the panel, and the proper overlay manager
is connected by omapdss.
This also allows us to add back the support for dynamic switching
between two exclusive panels. However, the current panel device model is
not changed to support this, as the new device model is implemented in
the following patches and the old model will be removed. The new device
model supports dynamic switching.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-05-08 17:23:32 +04:00
static int omapdss_default_connect ( struct omap_dss_device * dssdev )
{
OMAPDSS: combine omap_dss_output into omap_dss_device
We currently have omap_dss_device, which represents an external display
device, sometimes an external encoder, sometimes a panel. Then we have
omap_dss_output, which represents DSS's output encoder.
In the future with new display device model, we construct a video
pipeline from the display blocks. To accomplish this, all the blocks
need to be presented by the same entity.
Thus, this patch combines omap_dss_output into omap_dss_device. Some of
the fields in omap_dss_output are already found in omap_dss_device, but
some are not. This means we'll have DSS output specific fields in
omap_dss_device, which is not very nice. However, it is easier to just
keep those output specific fields there for now, and after transition to
new display device model is made, they can be cleaned up easier than
could be done now.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-04-19 16:09:34 +04:00
struct omap_dss_device * out ;
OMAPDSS: Implement display (dis)connect support
We currently have two steps in panel initialization and startup: probing
and enabling. After the panel has been probed, it's ready and can be
configured and later enabled.
This model is not enough with more complex display pipelines, where we
may have, for example, two panels, of which only one can be used at a
time, connected to the same video output.
To support that kind of scenarios, we need to add new step to the
initialization: connect.
This patch adds support for connecting and disconnecting panels. After
probe, but before connect, no panel ops should be called. When the
connect is called, a proper video pipeline is established, and the panel
is ready for use. If some part in the video pipeline is already
connected (by some other panel), the connect call fails.
One key difference with the old style setup is that connect() handles
also connecting to the overlay manager. This means that the omapfb (or
omapdrm) no longer needs to figure out which overlay manager to use, but
it can just call connect() on the panel, and the proper overlay manager
is connected by omapdss.
This also allows us to add back the support for dynamic switching
between two exclusive panels. However, the current panel device model is
not changed to support this, as the new device model is implemented in
the following patches and the old model will be removed. The new device
model supports dynamic switching.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-05-08 17:23:32 +04:00
struct omap_overlay_manager * mgr ;
int r ;
out = dssdev - > output ;
if ( out = = NULL )
return - ENODEV ;
mgr = omap_dss_get_overlay_manager ( out - > dispc_channel ) ;
if ( ! mgr )
return - ENODEV ;
r = dss_mgr_connect ( mgr , out ) ;
if ( r )
return r ;
return 0 ;
}
static void omapdss_default_disconnect ( struct omap_dss_device * dssdev )
{
OMAPDSS: combine omap_dss_output into omap_dss_device
We currently have omap_dss_device, which represents an external display
device, sometimes an external encoder, sometimes a panel. Then we have
omap_dss_output, which represents DSS's output encoder.
In the future with new display device model, we construct a video
pipeline from the display blocks. To accomplish this, all the blocks
need to be presented by the same entity.
Thus, this patch combines omap_dss_output into omap_dss_device. Some of
the fields in omap_dss_output are already found in omap_dss_device, but
some are not. This means we'll have DSS output specific fields in
omap_dss_device, which is not very nice. However, it is easier to just
keep those output specific fields there for now, and after transition to
new display device model is made, they can be cleaned up easier than
could be done now.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-04-19 16:09:34 +04:00
struct omap_dss_device * out ;
OMAPDSS: Implement display (dis)connect support
We currently have two steps in panel initialization and startup: probing
and enabling. After the panel has been probed, it's ready and can be
configured and later enabled.
This model is not enough with more complex display pipelines, where we
may have, for example, two panels, of which only one can be used at a
time, connected to the same video output.
To support that kind of scenarios, we need to add new step to the
initialization: connect.
This patch adds support for connecting and disconnecting panels. After
probe, but before connect, no panel ops should be called. When the
connect is called, a proper video pipeline is established, and the panel
is ready for use. If some part in the video pipeline is already
connected (by some other panel), the connect call fails.
One key difference with the old style setup is that connect() handles
also connecting to the overlay manager. This means that the omapfb (or
omapdrm) no longer needs to figure out which overlay manager to use, but
it can just call connect() on the panel, and the proper overlay manager
is connected by omapdss.
This also allows us to add back the support for dynamic switching
between two exclusive panels. However, the current panel device model is
not changed to support this, as the new device model is implemented in
the following patches and the old model will be removed. The new device
model supports dynamic switching.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-05-08 17:23:32 +04:00
struct omap_overlay_manager * mgr ;
out = dssdev - > output ;
if ( out = = NULL )
return ;
mgr = out - > manager ;
if ( mgr = = NULL )
return ;
dss_mgr_disconnect ( mgr , out ) ;
}
2009-11-03 12:23:50 +03:00
int omap_dss_register_driver ( struct omap_dss_driver * dssdriver )
{
dssdriver - > driver . bus = & dss_bus_type ;
dssdriver - > driver . probe = dss_driver_probe ;
dssdriver - > driver . remove = dss_driver_remove ;
2010-01-11 14:54:33 +03:00
if ( dssdriver - > get_resolution = = NULL )
dssdriver - > get_resolution = omapdss_default_get_resolution ;
2010-01-11 15:33:40 +03:00
if ( dssdriver - > get_recommended_bpp = = NULL )
dssdriver - > get_recommended_bpp =
omapdss_default_get_recommended_bpp ;
2012-03-15 22:00:23 +04:00
if ( dssdriver - > get_timings = = NULL )
dssdriver - > get_timings = omapdss_default_get_timings ;
OMAPDSS: Implement display (dis)connect support
We currently have two steps in panel initialization and startup: probing
and enabling. After the panel has been probed, it's ready and can be
configured and later enabled.
This model is not enough with more complex display pipelines, where we
may have, for example, two panels, of which only one can be used at a
time, connected to the same video output.
To support that kind of scenarios, we need to add new step to the
initialization: connect.
This patch adds support for connecting and disconnecting panels. After
probe, but before connect, no panel ops should be called. When the
connect is called, a proper video pipeline is established, and the panel
is ready for use. If some part in the video pipeline is already
connected (by some other panel), the connect call fails.
One key difference with the old style setup is that connect() handles
also connecting to the overlay manager. This means that the omapfb (or
omapdrm) no longer needs to figure out which overlay manager to use, but
it can just call connect() on the panel, and the proper overlay manager
is connected by omapdss.
This also allows us to add back the support for dynamic switching
between two exclusive panels. However, the current panel device model is
not changed to support this, as the new device model is implemented in
the following patches and the old model will be removed. The new device
model supports dynamic switching.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-05-08 17:23:32 +04:00
if ( dssdriver - > connect = = NULL )
dssdriver - > connect = omapdss_default_connect ;
if ( dssdriver - > disconnect = = NULL )
dssdriver - > disconnect = omapdss_default_disconnect ;
2010-01-11 14:54:33 +03:00
2009-11-03 12:23:50 +03:00
return driver_register ( & dssdriver - > driver ) ;
}
EXPORT_SYMBOL ( omap_dss_register_driver ) ;
void omap_dss_unregister_driver ( struct omap_dss_driver * dssdriver )
{
driver_unregister ( & dssdriver - > driver ) ;
}
EXPORT_SYMBOL ( omap_dss_unregister_driver ) ;
/* DEVICE */
static void omap_dss_dev_release ( struct device * dev )
{
2012-09-10 14:58:29 +04:00
struct omap_dss_device * dssdev = to_dss_device ( dev ) ;
kfree ( dssdev ) ;
2009-11-03 12:23:50 +03:00
}
2012-09-06 14:57:39 +04:00
static int disp_num_counter ;
2012-09-10 14:58:29 +04:00
struct omap_dss_device * dss_alloc_and_init_device ( struct device * parent )
2009-11-03 12:23:50 +03:00
{
2012-09-10 14:58:29 +04:00
struct omap_dss_device * dssdev ;
dssdev = kzalloc ( sizeof ( * dssdev ) , GFP_KERNEL ) ;
if ( ! dssdev )
return NULL ;
2012-09-06 14:57:39 +04:00
OMAPDSS: Add panel dev pointer to dssdev
We are about to remove the dss bus support, which also means that the
omap_dss_device won't be a real device anymore. This means that the
embedded "dev" struct needs to be removed from omap_dss_device.
After we've finished the removal of the dss bus, we see the following
changes:
- struct omap_dss_device won't be a real Linux device anymore, but more
like a "display entity".
- struct omap_dss_driver won't be a Linux device driver, but "display
entity ops".
- The panel devices/drivers won't be omapdss devices/drivers, but
platform/i2c/spi/etc devices/drivers, whichever fits the control
mechanism of the panel.
- The panel drivers will create omap_dss_device and omap_dss_driver,
fill the required fields, and register the omap_dss_device to
omapdss.
- omap_dss_device won't have an embedded dev struct anymore, but a
dev pointer to the actual device that manages the omap_dss_device.
The model described above resembles the model that has been discussed
with CDF (common display framework).
For the duration of the conversion, we temporarily have two devs in the
dssdev, the old "old_dev", which is a full embedded device struct, and the
new "dev", which is a pointer to the device. "old_dev" will be removed
in the future.
For devices belonging to dss bus the dev is initialized to point to
old_dev. This way all the code can just use the dev, for both old and
new style panels.
Both the new and old style panel drivers work during the conversion, and
only after the dss bus support is removed will the old style panels stop
to compile.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-02-14 16:17:28 +04:00
dssdev - > old_dev . bus = & dss_bus_type ;
dssdev - > old_dev . parent = parent ;
dssdev - > old_dev . release = omap_dss_dev_release ;
dev_set_name ( & dssdev - > old_dev , " display%d " , disp_num_counter + + ) ;
2012-09-10 14:58:29 +04:00
OMAPDSS: Add panel dev pointer to dssdev
We are about to remove the dss bus support, which also means that the
omap_dss_device won't be a real device anymore. This means that the
embedded "dev" struct needs to be removed from omap_dss_device.
After we've finished the removal of the dss bus, we see the following
changes:
- struct omap_dss_device won't be a real Linux device anymore, but more
like a "display entity".
- struct omap_dss_driver won't be a Linux device driver, but "display
entity ops".
- The panel devices/drivers won't be omapdss devices/drivers, but
platform/i2c/spi/etc devices/drivers, whichever fits the control
mechanism of the panel.
- The panel drivers will create omap_dss_device and omap_dss_driver,
fill the required fields, and register the omap_dss_device to
omapdss.
- omap_dss_device won't have an embedded dev struct anymore, but a
dev pointer to the actual device that manages the omap_dss_device.
The model described above resembles the model that has been discussed
with CDF (common display framework).
For the duration of the conversion, we temporarily have two devs in the
dssdev, the old "old_dev", which is a full embedded device struct, and the
new "dev", which is a pointer to the device. "old_dev" will be removed
in the future.
For devices belonging to dss bus the dev is initialized to point to
old_dev. This way all the code can just use the dev, for both old and
new style panels.
Both the new and old style panel drivers work during the conversion, and
only after the dss bus support is removed will the old style panels stop
to compile.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-02-14 16:17:28 +04:00
device_initialize ( & dssdev - > old_dev ) ;
2012-09-10 14:58:29 +04:00
return dssdev ;
2009-11-03 12:23:50 +03:00
}
2012-09-10 14:58:29 +04:00
int dss_add_device ( struct omap_dss_device * dssdev )
{
OMAPDSS: Add panel dev pointer to dssdev
We are about to remove the dss bus support, which also means that the
omap_dss_device won't be a real device anymore. This means that the
embedded "dev" struct needs to be removed from omap_dss_device.
After we've finished the removal of the dss bus, we see the following
changes:
- struct omap_dss_device won't be a real Linux device anymore, but more
like a "display entity".
- struct omap_dss_driver won't be a Linux device driver, but "display
entity ops".
- The panel devices/drivers won't be omapdss devices/drivers, but
platform/i2c/spi/etc devices/drivers, whichever fits the control
mechanism of the panel.
- The panel drivers will create omap_dss_device and omap_dss_driver,
fill the required fields, and register the omap_dss_device to
omapdss.
- omap_dss_device won't have an embedded dev struct anymore, but a
dev pointer to the actual device that manages the omap_dss_device.
The model described above resembles the model that has been discussed
with CDF (common display framework).
For the duration of the conversion, we temporarily have two devs in the
dssdev, the old "old_dev", which is a full embedded device struct, and the
new "dev", which is a pointer to the device. "old_dev" will be removed
in the future.
For devices belonging to dss bus the dev is initialized to point to
old_dev. This way all the code can just use the dev, for both old and
new style panels.
Both the new and old style panel drivers work during the conversion, and
only after the dss bus support is removed will the old style panels stop
to compile.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-02-14 16:17:28 +04:00
dssdev - > dev = & dssdev - > old_dev ;
2012-11-16 17:45:26 +04:00
omapdss_register_display ( dssdev ) ;
OMAPDSS: Add panel dev pointer to dssdev
We are about to remove the dss bus support, which also means that the
omap_dss_device won't be a real device anymore. This means that the
embedded "dev" struct needs to be removed from omap_dss_device.
After we've finished the removal of the dss bus, we see the following
changes:
- struct omap_dss_device won't be a real Linux device anymore, but more
like a "display entity".
- struct omap_dss_driver won't be a Linux device driver, but "display
entity ops".
- The panel devices/drivers won't be omapdss devices/drivers, but
platform/i2c/spi/etc devices/drivers, whichever fits the control
mechanism of the panel.
- The panel drivers will create omap_dss_device and omap_dss_driver,
fill the required fields, and register the omap_dss_device to
omapdss.
- omap_dss_device won't have an embedded dev struct anymore, but a
dev pointer to the actual device that manages the omap_dss_device.
The model described above resembles the model that has been discussed
with CDF (common display framework).
For the duration of the conversion, we temporarily have two devs in the
dssdev, the old "old_dev", which is a full embedded device struct, and the
new "dev", which is a pointer to the device. "old_dev" will be removed
in the future.
For devices belonging to dss bus the dev is initialized to point to
old_dev. This way all the code can just use the dev, for both old and
new style panels.
Both the new and old style panel drivers work during the conversion, and
only after the dss bus support is removed will the old style panels stop
to compile.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-02-14 16:17:28 +04:00
return device_add ( & dssdev - > old_dev ) ;
2012-09-10 14:58:29 +04:00
}
void dss_put_device ( struct omap_dss_device * dssdev )
{
OMAPDSS: Add panel dev pointer to dssdev
We are about to remove the dss bus support, which also means that the
omap_dss_device won't be a real device anymore. This means that the
embedded "dev" struct needs to be removed from omap_dss_device.
After we've finished the removal of the dss bus, we see the following
changes:
- struct omap_dss_device won't be a real Linux device anymore, but more
like a "display entity".
- struct omap_dss_driver won't be a Linux device driver, but "display
entity ops".
- The panel devices/drivers won't be omapdss devices/drivers, but
platform/i2c/spi/etc devices/drivers, whichever fits the control
mechanism of the panel.
- The panel drivers will create omap_dss_device and omap_dss_driver,
fill the required fields, and register the omap_dss_device to
omapdss.
- omap_dss_device won't have an embedded dev struct anymore, but a
dev pointer to the actual device that manages the omap_dss_device.
The model described above resembles the model that has been discussed
with CDF (common display framework).
For the duration of the conversion, we temporarily have two devs in the
dssdev, the old "old_dev", which is a full embedded device struct, and the
new "dev", which is a pointer to the device. "old_dev" will be removed
in the future.
For devices belonging to dss bus the dev is initialized to point to
old_dev. This way all the code can just use the dev, for both old and
new style panels.
Both the new and old style panel drivers work during the conversion, and
only after the dss bus support is removed will the old style panels stop
to compile.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-02-14 16:17:28 +04:00
put_device ( & dssdev - > old_dev ) ;
2012-09-10 14:58:29 +04:00
}
void dss_unregister_device ( struct omap_dss_device * dssdev )
2009-11-03 12:23:50 +03:00
{
OMAPDSS: Add panel dev pointer to dssdev
We are about to remove the dss bus support, which also means that the
omap_dss_device won't be a real device anymore. This means that the
embedded "dev" struct needs to be removed from omap_dss_device.
After we've finished the removal of the dss bus, we see the following
changes:
- struct omap_dss_device won't be a real Linux device anymore, but more
like a "display entity".
- struct omap_dss_driver won't be a Linux device driver, but "display
entity ops".
- The panel devices/drivers won't be omapdss devices/drivers, but
platform/i2c/spi/etc devices/drivers, whichever fits the control
mechanism of the panel.
- The panel drivers will create omap_dss_device and omap_dss_driver,
fill the required fields, and register the omap_dss_device to
omapdss.
- omap_dss_device won't have an embedded dev struct anymore, but a
dev pointer to the actual device that manages the omap_dss_device.
The model described above resembles the model that has been discussed
with CDF (common display framework).
For the duration of the conversion, we temporarily have two devs in the
dssdev, the old "old_dev", which is a full embedded device struct, and the
new "dev", which is a pointer to the device. "old_dev" will be removed
in the future.
For devices belonging to dss bus the dev is initialized to point to
old_dev. This way all the code can just use the dev, for both old and
new style panels.
Both the new and old style panel drivers work during the conversion, and
only after the dss bus support is removed will the old style panels stop
to compile.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2013-02-14 16:17:28 +04:00
device_unregister ( & dssdev - > old_dev ) ;
2012-11-16 17:45:26 +04:00
omapdss_unregister_display ( dssdev ) ;
2009-11-03 12:23:50 +03:00
}
OMAPDSS: interface drivers register their panel devices
Currently the higher level omapdss platform driver gets the list of
displays in its platform data, and uses that list to create the
omap_dss_device for each display.
With DT, the logical way to do the above is to list the displays under
each individual output, i.e. we'd have "dpi" node, under which we would
have the display that uses DPI. In other words, each output driver
handles the displays that use that particular output.
To make the current code ready for DT, this patch modifies the output
drivers so that each of them creates the display devices which use that
output. However, instead of changing the platform data to suit this
method, each output driver is passed the full list of displays, and the
drivers pick the displays that are meant for them. This allows us to
keep the old platform data, and thus we avoid the need to change the
board files.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-03-01 17:45:53 +04:00
static int dss_unregister_dss_dev ( struct device * dev , void * data )
{
struct omap_dss_device * dssdev = to_dss_device ( dev ) ;
2012-09-10 14:58:29 +04:00
dss_unregister_device ( dssdev ) ;
OMAPDSS: interface drivers register their panel devices
Currently the higher level omapdss platform driver gets the list of
displays in its platform data, and uses that list to create the
omap_dss_device for each display.
With DT, the logical way to do the above is to list the displays under
each individual output, i.e. we'd have "dpi" node, under which we would
have the display that uses DPI. In other words, each output driver
handles the displays that use that particular output.
To make the current code ready for DT, this patch modifies the output
drivers so that each of them creates the display devices which use that
output. However, instead of changing the platform data to suit this
method, each output driver is passed the full list of displays, and the
drivers pick the displays that are meant for them. This allows us to
keep the old platform data, and thus we avoid the need to change the
board files.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-03-01 17:45:53 +04:00
return 0 ;
}
2012-09-10 14:58:29 +04:00
void dss_unregister_child_devices ( struct device * parent )
OMAPDSS: interface drivers register their panel devices
Currently the higher level omapdss platform driver gets the list of
displays in its platform data, and uses that list to create the
omap_dss_device for each display.
With DT, the logical way to do the above is to list the displays under
each individual output, i.e. we'd have "dpi" node, under which we would
have the display that uses DPI. In other words, each output driver
handles the displays that use that particular output.
To make the current code ready for DT, this patch modifies the output
drivers so that each of them creates the display devices which use that
output. However, instead of changing the platform data to suit this
method, each output driver is passed the full list of displays, and the
drivers pick the displays that are meant for them. This allows us to
keep the old platform data, and thus we avoid the need to change the
board files.
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
2012-03-01 17:45:53 +04:00
{
device_for_each_child ( parent , NULL , dss_unregister_dss_dev ) ;
}
2012-09-10 14:58:29 +04:00
void dss_copy_device_pdata ( struct omap_dss_device * dst ,
const struct omap_dss_device * src )
{
u8 * d = ( u8 * ) dst ;
u8 * s = ( u8 * ) src ;
size_t dsize = sizeof ( struct device ) ;
memcpy ( d + dsize , s + dsize , sizeof ( struct omap_dss_device ) - dsize ) ;
}
2009-11-03 12:23:50 +03:00
/* BUS */
2012-02-17 19:41:13 +04:00
static int __init omap_dss_bus_register ( void )
2009-11-03 12:23:50 +03:00
{
int r ;
r = bus_register ( & dss_bus_type ) ;
if ( r ) {
DSSERR ( " bus register failed \n " ) ;
return r ;
}
dev_set_name ( & dss_bus , " omapdss " ) ;
r = device_register ( & dss_bus ) ;
if ( r ) {
DSSERR ( " bus driver register failed \n " ) ;
bus_unregister ( & dss_bus_type ) ;
return r ;
}
return 0 ;
}
/* INIT */
2012-03-02 19:37:53 +04:00
static int ( * dss_output_drv_reg_funcs [ ] ) ( void ) __initdata = {
2012-10-22 16:57:25 +04:00
# ifdef CONFIG_OMAP2_DSS_DSI
dsi_init_platform_driver ,
# endif
2012-03-02 19:37:53 +04:00
# ifdef CONFIG_OMAP2_DSS_DPI
dpi_init_platform_driver ,
# endif
# ifdef CONFIG_OMAP2_DSS_SDI
sdi_init_platform_driver ,
# endif
# ifdef CONFIG_OMAP2_DSS_RFBI
rfbi_init_platform_driver ,
# endif
# ifdef CONFIG_OMAP2_DSS_VENC
venc_init_platform_driver ,
# endif
# ifdef CONFIG_OMAP4_DSS_HDMI
hdmi_init_platform_driver ,
# endif
} ;
static void ( * dss_output_drv_unreg_funcs [ ] ) ( void ) __exitdata = {
2012-10-22 16:57:25 +04:00
# ifdef CONFIG_OMAP2_DSS_DSI
dsi_uninit_platform_driver ,
# endif
2012-03-02 19:37:53 +04:00
# ifdef CONFIG_OMAP2_DSS_DPI
dpi_uninit_platform_driver ,
# endif
# ifdef CONFIG_OMAP2_DSS_SDI
sdi_uninit_platform_driver ,
# endif
# ifdef CONFIG_OMAP2_DSS_RFBI
rfbi_uninit_platform_driver ,
# endif
# ifdef CONFIG_OMAP2_DSS_VENC
venc_uninit_platform_driver ,
# endif
# ifdef CONFIG_OMAP4_DSS_HDMI
hdmi_uninit_platform_driver ,
# endif
} ;
static bool dss_output_drv_loaded [ ARRAY_SIZE ( dss_output_drv_reg_funcs ) ] ;
2009-11-03 12:23:50 +03:00
2012-03-19 17:05:02 +04:00
static int __init omap_dss_register_drivers ( void )
{
int r ;
2012-03-02 19:37:53 +04:00
int i ;
2012-03-19 17:05:02 +04:00
2012-03-07 14:53:18 +04:00
r = platform_driver_probe ( & omap_dss_driver , omap_dss_probe ) ;
2012-03-19 17:05:02 +04:00
if ( r )
return r ;
r = dss_init_platform_driver ( ) ;
if ( r ) {
DSSERR ( " Failed to initialize DSS platform driver \n " ) ;
goto err_dss ;
}
r = dispc_init_platform_driver ( ) ;
if ( r ) {
DSSERR ( " Failed to initialize dispc platform driver \n " ) ;
goto err_dispc ;
}
2012-03-02 19:37:53 +04:00
/*
* It ' s ok if the output - driver register fails . It happens , for example ,
* when there is no output - device ( e . g . SDI for OMAP4 ) .
*/
for ( i = 0 ; i < ARRAY_SIZE ( dss_output_drv_reg_funcs ) ; + + i ) {
r = dss_output_drv_reg_funcs [ i ] ( ) ;
if ( r = = 0 )
dss_output_drv_loaded [ i ] = true ;
2012-03-19 17:05:02 +04:00
}
return 0 ;
err_dispc :
dss_uninit_platform_driver ( ) ;
err_dss :
platform_driver_unregister ( & omap_dss_driver ) ;
return r ;
}
static void __exit omap_dss_unregister_drivers ( void )
{
2012-03-02 19:37:53 +04:00
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( dss_output_drv_unreg_funcs ) ; + + i ) {
if ( dss_output_drv_loaded [ i ] )
dss_output_drv_unreg_funcs [ i ] ( ) ;
}
2012-03-19 17:05:02 +04:00
dispc_uninit_platform_driver ( ) ;
dss_uninit_platform_driver ( ) ;
platform_driver_unregister ( & omap_dss_driver ) ;
}
2009-11-03 12:23:50 +03:00
# ifdef CONFIG_OMAP2_DSS_MODULE
static void omap_dss_bus_unregister ( void )
{
device_unregister ( & dss_bus ) ;
bus_unregister ( & dss_bus_type ) ;
}
static int __init omap_dss_init ( void )
{
int r ;
r = omap_dss_bus_register ( ) ;
if ( r )
return r ;
2012-03-19 17:05:02 +04:00
r = omap_dss_register_drivers ( ) ;
2009-11-03 12:23:50 +03:00
if ( r ) {
omap_dss_bus_unregister ( ) ;
return r ;
}
2013-05-23 13:07:50 +04:00
dss_initialized = true ;
2009-11-03 12:23:50 +03:00
return 0 ;
}
static void __exit omap_dss_exit ( void )
{
2012-03-19 17:05:02 +04:00
omap_dss_unregister_drivers ( ) ;
2009-11-03 12:23:50 +03:00
omap_dss_bus_unregister ( ) ;
}
module_init ( omap_dss_init ) ;
module_exit ( omap_dss_exit ) ;
# else
static int __init omap_dss_init ( void )
{
return omap_dss_bus_register ( ) ;
}
static int __init omap_dss_init2 ( void )
{
2013-05-23 13:07:50 +04:00
int r ;
r = omap_dss_register_drivers ( ) ;
if ( r )
return r ;
dss_initialized = true ;
return 0 ;
2009-11-03 12:23:50 +03:00
}
core_initcall ( omap_dss_init ) ;
device_initcall ( omap_dss_init2 ) ;
# endif
MODULE_AUTHOR ( " Tomi Valkeinen <tomi.valkeinen@nokia.com> " ) ;
MODULE_DESCRIPTION ( " OMAP2/3 Display Subsystem " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;