2012-10-04 17:32:52 +04:00
/*
* OF helpers for parsing display timings
*
* Copyright ( c ) 2012 Steffen Trumtrar < s . trumtrar @ pengutronix . de > , Pengutronix
*
* based on of_videomode . c by Sascha Hauer < s . hauer @ pengutronix . de >
*
* This file is released under the GPLv2
*/
# include <linux/export.h>
# include <linux/of.h>
# include <linux/slab.h>
# include <video/display_timing.h>
# include <video/of_display_timing.h>
/**
* parse_timing_property - parse timing_entry from device_node
* @ np : device_node with the property
* @ name : name of the property
* @ result : will be set to the return value
*
* DESCRIPTION :
* Every display_timing can be specified with either just the typical value or
* a range consisting of min / typ / max . This function helps handling this
* */
2013-05-27 16:33:05 +04:00
static int parse_timing_property ( const struct device_node * np , const char * name ,
2012-10-04 17:32:52 +04:00
struct timing_entry * result )
{
struct property * prop ;
int length , cells , ret ;
prop = of_find_property ( np , name , & length ) ;
if ( ! prop ) {
pr_err ( " %s: could not find property %s \n " ,
of_node_full_name ( np ) , name ) ;
return - EINVAL ;
}
cells = length / sizeof ( u32 ) ;
if ( cells = = 1 ) {
ret = of_property_read_u32 ( np , name , & result - > typ ) ;
result - > min = result - > typ ;
result - > max = result - > typ ;
} else if ( cells = = 3 ) {
ret = of_property_read_u32_array ( np , name , & result - > min , cells ) ;
} else {
pr_err ( " %s: illegal timing specification in %s \n " ,
of_node_full_name ( np ) , name ) ;
return - EINVAL ;
}
return ret ;
}
/**
2013-05-16 16:36:38 +04:00
* of_parse_display_timing - parse display_timing entry from device_node
2012-10-04 17:32:52 +04:00
* @ np : device_node with the properties
* */
Merge branch 'drm-next' of git://people.freedesktop.org/~airlied/linux
Pull drm updates from Dave Airlie:
"Okay this is the big one, I was stalled on the fbdev pull req as I
stupidly let fbdev guys merge a patch I required to fix a warning with
some patches I had, they ended up merging the patch from the wrong
place, but the warning should be fixed. In future I'll just take the
patch myself!
Outside drm:
There are some snd changes for the HDMI audio interactions on haswell,
they've been acked for inclusion via my tree. This relies on the
wound/wait tree from Ingo which is already merged.
Major changes:
AMD finally released the dynamic power management code for all their
GPUs from r600->present day, this is great, off by default for now but
also a huge amount of code, in fact it is most of this pull request.
Since it landed there has been a lot of community testing and Alex has
sent a lot of fixes for any bugs found so far. I suspect radeon might
now be the biggest kernel driver ever :-P p.s. radeon.dpm=1 to enable
dynamic powermanagement for anyone.
New drivers:
Renesas r-car display unit.
Other highlights:
- core: GEM CMA prime support, use new w/w mutexs for TTM
reservations, cursor hotspot, doc updates
- dvo chips: chrontel 7010B support
- i915: Haswell (fbc, ips, vecs, watermarks, audio powerwell),
Valleyview (enabled by default, rc6), lots of pll reworking, 30bpp
support (this time for sure)
- nouveau: async buffer object deletion, context/register init
updates, kernel vp2 engine support, GF117 support, GK110 accel
support (with external nvidia ucode), context cleanups.
- exynos: memory leak fixes, Add S3C64XX SoC series support, device
tree updates, common clock framework support,
- qxl: cursor hotspot support, multi-monitor support, suspend/resume
support
- mgag200: hw cursor support, g200 mode limiting
- shmobile: prime support
- tegra: fixes mostly
I've been banging on this quite a lot due to the size of it, and it
seems to okay on everything I've tested it on."
* 'drm-next' of git://people.freedesktop.org/~airlied/linux: (811 commits)
drm/radeon/dpm: implement vblank_too_short callback for si
drm/radeon/dpm: implement vblank_too_short callback for cayman
drm/radeon/dpm: implement vblank_too_short callback for btc
drm/radeon/dpm: implement vblank_too_short callback for evergreen
drm/radeon/dpm: implement vblank_too_short callback for 7xx
drm/radeon/dpm: add checks against vblank time
drm/radeon/dpm: add helper to calculate vblank time
drm/radeon: remove stray line in old pm code
drm/radeon/dpm: fix display_gap programming on rv7xx
drm/nvc0/gr: fix gpc firmware regression
drm/nouveau: fix minor thinko causing bo moves to not be async on kepler
drm/radeon/dpm: implement force performance level for TN
drm/radeon/dpm: implement force performance level for ON/LN
drm/radeon/dpm: implement force performance level for SI
drm/radeon/dpm: implement force performance level for cayman
drm/radeon/dpm: implement force performance levels for 7xx/eg/btc
drm/radeon/dpm: add infrastructure to force performance levels
drm/radeon: fix surface setup on r1xx
drm/radeon: add support for 3d perf states on older asics
drm/radeon: set default clocks for SI when DPM is disabled
...
2013-07-10 03:04:31 +04:00
static int of_parse_display_timing ( const struct device_node * np ,
2013-05-16 16:29:06 +04:00
struct display_timing * dt )
2012-10-04 17:32:52 +04:00
{
u32 val = 0 ;
int ret = 0 ;
2013-05-16 16:29:06 +04:00
memset ( dt , 0 , sizeof ( * dt ) ) ;
2012-10-04 17:32:52 +04:00
ret | = parse_timing_property ( np , " hback-porch " , & dt - > hback_porch ) ;
ret | = parse_timing_property ( np , " hfront-porch " , & dt - > hfront_porch ) ;
ret | = parse_timing_property ( np , " hactive " , & dt - > hactive ) ;
ret | = parse_timing_property ( np , " hsync-len " , & dt - > hsync_len ) ;
ret | = parse_timing_property ( np , " vback-porch " , & dt - > vback_porch ) ;
ret | = parse_timing_property ( np , " vfront-porch " , & dt - > vfront_porch ) ;
ret | = parse_timing_property ( np , " vactive " , & dt - > vactive ) ;
ret | = parse_timing_property ( np , " vsync-len " , & dt - > vsync_len ) ;
ret | = parse_timing_property ( np , " clock-frequency " , & dt - > pixelclock ) ;
2013-03-12 12:26:45 +04:00
dt - > flags = 0 ;
2012-10-04 17:32:52 +04:00
if ( ! of_property_read_u32 ( np , " vsync-active " , & val ) )
2013-03-12 12:26:45 +04:00
dt - > flags | = val ? DISPLAY_FLAGS_VSYNC_HIGH :
DISPLAY_FLAGS_VSYNC_LOW ;
2012-10-04 17:32:52 +04:00
if ( ! of_property_read_u32 ( np , " hsync-active " , & val ) )
2013-03-12 12:26:45 +04:00
dt - > flags | = val ? DISPLAY_FLAGS_HSYNC_HIGH :
DISPLAY_FLAGS_HSYNC_LOW ;
2012-10-04 17:32:52 +04:00
if ( ! of_property_read_u32 ( np , " de-active " , & val ) )
2013-03-12 12:26:45 +04:00
dt - > flags | = val ? DISPLAY_FLAGS_DE_HIGH :
2012-10-04 17:32:52 +04:00
DISPLAY_FLAGS_DE_LOW ;
if ( ! of_property_read_u32 ( np , " pixelclk-active " , & val ) )
2013-03-12 12:26:45 +04:00
dt - > flags | = val ? DISPLAY_FLAGS_PIXDATA_POSEDGE :
2012-10-04 17:32:52 +04:00
DISPLAY_FLAGS_PIXDATA_NEGEDGE ;
if ( of_property_read_bool ( np , " interlaced " ) )
2013-03-12 12:26:45 +04:00
dt - > flags | = DISPLAY_FLAGS_INTERLACED ;
2012-10-04 17:32:52 +04:00
if ( of_property_read_bool ( np , " doublescan " ) )
2013-03-12 12:26:45 +04:00
dt - > flags | = DISPLAY_FLAGS_DOUBLESCAN ;
2013-05-27 16:33:34 +04:00
if ( of_property_read_bool ( np , " doubleclk " ) )
dt - > flags | = DISPLAY_FLAGS_DOUBLECLK ;
2012-10-04 17:32:52 +04:00
if ( ret ) {
pr_err ( " %s: error reading timing properties \n " ,
of_node_full_name ( np ) ) ;
2013-05-16 16:29:06 +04:00
return - EINVAL ;
2012-10-04 17:32:52 +04:00
}
2013-05-16 16:29:06 +04:00
return 0 ;
2012-10-04 17:32:52 +04:00
}
2013-05-16 16:36:38 +04:00
/**
* of_get_display_timing - parse a display_timing entry
* @ np : device_node with the timing subnode
* @ name : name of the timing node
* @ dt : display_timing struct to fill
* */
int of_get_display_timing ( struct device_node * np , const char * name ,
struct display_timing * dt )
{
struct device_node * timing_np ;
2014-05-13 16:58:21 +04:00
if ( ! np )
2013-05-16 16:36:38 +04:00
return - EINVAL ;
2013-09-25 15:51:31 +04:00
timing_np = of_get_child_by_name ( np , name ) ;
2013-05-16 16:36:38 +04:00
if ( ! timing_np ) {
pr_err ( " %s: could not find node '%s' \n " ,
of_node_full_name ( np ) , name ) ;
return - ENOENT ;
}
return of_parse_display_timing ( timing_np , dt ) ;
}
EXPORT_SYMBOL_GPL ( of_get_display_timing ) ;
2012-10-04 17:32:52 +04:00
/**
* of_get_display_timings - parse all display_timing entries from a device_node
* @ np : device_node with the subnodes
* */
struct display_timings * of_get_display_timings ( struct device_node * np )
{
struct device_node * timings_np ;
struct device_node * entry ;
struct device_node * native_mode ;
struct display_timings * disp ;
2014-05-13 16:58:21 +04:00
if ( ! np )
2012-10-04 17:32:52 +04:00
return NULL ;
2013-09-25 15:51:31 +04:00
timings_np = of_get_child_by_name ( np , " display-timings " ) ;
2012-10-04 17:32:52 +04:00
if ( ! timings_np ) {
pr_err ( " %s: could not find display-timings node \n " ,
of_node_full_name ( np ) ) ;
return NULL ;
}
disp = kzalloc ( sizeof ( * disp ) , GFP_KERNEL ) ;
if ( ! disp ) {
pr_err ( " %s: could not allocate struct disp' \n " ,
of_node_full_name ( np ) ) ;
goto dispfail ;
}
entry = of_parse_phandle ( timings_np , " native-mode " , 0 ) ;
/* assume first child as native mode if none provided */
if ( ! entry )
2014-05-09 17:53:12 +04:00
entry = of_get_next_child ( timings_np , NULL ) ;
2012-10-04 17:32:52 +04:00
/* if there is no child, it is useless to go on */
if ( ! entry ) {
pr_err ( " %s: no timing specifications given \n " ,
of_node_full_name ( np ) ) ;
goto entryfail ;
}
pr_debug ( " %s: using %s as default timing \n " ,
of_node_full_name ( np ) , entry - > name ) ;
native_mode = entry ;
disp - > num_timings = of_get_child_count ( timings_np ) ;
if ( disp - > num_timings = = 0 ) {
/* should never happen, as entry was already found above */
pr_err ( " %s: no timings specified \n " , of_node_full_name ( np ) ) ;
goto entryfail ;
}
disp - > timings = kzalloc ( sizeof ( struct display_timing * ) *
disp - > num_timings , GFP_KERNEL ) ;
if ( ! disp - > timings ) {
pr_err ( " %s: could not allocate timings array \n " ,
of_node_full_name ( np ) ) ;
goto entryfail ;
}
disp - > num_timings = 0 ;
disp - > native_mode = 0 ;
for_each_child_of_node ( timings_np , entry ) {
struct display_timing * dt ;
2013-05-16 16:29:06 +04:00
int r ;
2012-10-04 17:32:52 +04:00
2013-05-16 16:29:06 +04:00
dt = kzalloc ( sizeof ( * dt ) , GFP_KERNEL ) ;
2012-10-04 17:32:52 +04:00
if ( ! dt ) {
2013-05-16 16:29:06 +04:00
pr_err ( " %s: could not allocate display_timing struct \n " ,
of_node_full_name ( np ) ) ;
goto timingfail ;
}
2013-05-16 16:36:38 +04:00
r = of_parse_display_timing ( entry , dt ) ;
2013-05-16 16:29:06 +04:00
if ( r ) {
2012-10-04 17:32:52 +04:00
/*
* to not encourage wrong devicetrees , fail in case of
* an error
*/
pr_err ( " %s: error in timing %d \n " ,
of_node_full_name ( np ) , disp - > num_timings + 1 ) ;
goto timingfail ;
}
if ( native_mode = = entry )
disp - > native_mode = disp - > num_timings ;
disp - > timings [ disp - > num_timings ] = dt ;
disp - > num_timings + + ;
}
of_node_put ( timings_np ) ;
/*
* native_mode points to the device_node returned by of_parse_phandle
* therefore call of_node_put on it
*/
of_node_put ( native_mode ) ;
pr_debug ( " %s: got %d timings. Using timing #%d as default \n " ,
of_node_full_name ( np ) , disp - > num_timings ,
disp - > native_mode + 1 ) ;
return disp ;
timingfail :
2014-08-08 14:07:55 +04:00
of_node_put ( native_mode ) ;
2012-10-04 17:32:52 +04:00
display_timings_release ( disp ) ;
2014-07-11 13:21:36 +04:00
disp = NULL ;
2012-10-04 17:32:52 +04:00
entryfail :
kfree ( disp ) ;
dispfail :
of_node_put ( timings_np ) ;
return NULL ;
}
EXPORT_SYMBOL_GPL ( of_get_display_timings ) ;
/**
* of_display_timings_exist - check if a display - timings node is provided
* @ np : device_node with the timing
* */
int of_display_timings_exist ( struct device_node * np )
{
struct device_node * timings_np ;
if ( ! np )
return - EINVAL ;
timings_np = of_parse_phandle ( np , " display-timings " , 0 ) ;
if ( ! timings_np )
return - EINVAL ;
of_node_put ( timings_np ) ;
return 1 ;
}
EXPORT_SYMBOL_GPL ( of_display_timings_exist ) ;