2017-12-14 08:49:35 +03:00
// SPDX-License-Identifier: (GPL-2.0 OR BSD-3-Clause)
// Copyright(c) 2015-17 Intel Corporation.
/*
* MIPI Discovery And Configuration ( DisCo ) Specification for SoundWire
* specifies properties to be implemented for SoundWire Masters and Slaves .
* The DisCo spec doesn ' t mandate these properties . However , SDW bus cannot
* work without knowing these values .
*
* The helper functions read the Master and Slave properties . Implementers
* of Master or Slave drivers can use any of the below three mechanisms :
* a ) Use these APIs here as . read_prop ( ) callback for Master and Slave
* b ) Implement own methods and set those as . read_prop ( ) , but invoke
* APIs in this file for generic read and override the values with
* platform specific data
* c ) Implement ones own methods which do not use anything provided
* here
*/
# include <linux/device.h>
# include <linux/property.h>
# include <linux/mod_devicetable.h>
# include <linux/soundwire/sdw.h>
# include "bus.h"
/**
* sdw_master_read_prop ( ) - Read Master properties
* @ bus : SDW bus instance
*/
int sdw_master_read_prop ( struct sdw_bus * bus )
{
struct sdw_master_prop * prop = & bus - > prop ;
struct fwnode_handle * link ;
char name [ 32 ] ;
int nval , i ;
device_property_read_u32 ( bus - > dev ,
2019-05-01 18:57:33 +03:00
" mipi-sdw-sw-interface-revision " ,
& prop - > revision ) ;
2017-12-14 08:49:35 +03:00
/* Find master handle */
snprintf ( name , sizeof ( name ) ,
2019-05-22 22:47:18 +03:00
" mipi-sdw-link-%d-subproperties " , bus - > link_id ) ;
2017-12-14 08:49:35 +03:00
link = device_get_named_child_node ( bus - > dev , name ) ;
if ( ! link ) {
dev_err ( bus - > dev , " Master node %s not found \n " , name ) ;
return - EIO ;
}
if ( fwnode_property_read_bool ( link ,
2019-05-01 18:57:34 +03:00
" mipi-sdw-clock-stop-mode0-supported " ) )
2017-12-14 08:49:35 +03:00
prop - > clk_stop_mode = SDW_CLK_STOP_MODE0 ;
if ( fwnode_property_read_bool ( link ,
2019-05-01 18:57:34 +03:00
" mipi-sdw-clock-stop-mode1-supported " ) )
2017-12-14 08:49:35 +03:00
prop - > clk_stop_mode | = SDW_CLK_STOP_MODE1 ;
fwnode_property_read_u32 ( link ,
2019-05-01 18:57:33 +03:00
" mipi-sdw-max-clock-frequency " ,
2019-05-22 22:47:22 +03:00
& prop - > max_clk_freq ) ;
2017-12-14 08:49:35 +03:00
nval = fwnode_property_read_u32_array ( link ,
" mipi-sdw-clock-frequencies-supported " , NULL , 0 ) ;
if ( nval > 0 ) {
2019-05-22 22:47:22 +03:00
prop - > num_clk_freq = nval ;
prop - > clk_freq = devm_kcalloc ( bus - > dev , prop - > num_clk_freq ,
sizeof ( * prop - > clk_freq ) ,
GFP_KERNEL ) ;
if ( ! prop - > clk_freq )
2017-12-14 08:49:35 +03:00
return - ENOMEM ;
fwnode_property_read_u32_array ( link ,
" mipi-sdw-clock-frequencies-supported " ,
2019-05-22 22:47:22 +03:00
prop - > clk_freq , prop - > num_clk_freq ) ;
2017-12-14 08:49:35 +03:00
}
/*
* Check the frequencies supported . If FW doesn ' t provide max
* freq , then populate here by checking values .
*/
2019-05-22 22:47:22 +03:00
if ( ! prop - > max_clk_freq & & prop - > clk_freq ) {
prop - > max_clk_freq = prop - > clk_freq [ 0 ] ;
for ( i = 1 ; i < prop - > num_clk_freq ; i + + ) {
if ( prop - > clk_freq [ i ] > prop - > max_clk_freq )
prop - > max_clk_freq = prop - > clk_freq [ i ] ;
2017-12-14 08:49:35 +03:00
}
}
nval = fwnode_property_read_u32_array ( link ,
" mipi-sdw-supported-clock-gears " , NULL , 0 ) ;
if ( nval > 0 ) {
prop - > num_clk_gears = nval ;
prop - > clk_gears = devm_kcalloc ( bus - > dev , prop - > num_clk_gears ,
2019-05-01 18:57:33 +03:00
sizeof ( * prop - > clk_gears ) ,
GFP_KERNEL ) ;
2017-12-14 08:49:35 +03:00
if ( ! prop - > clk_gears )
return - ENOMEM ;
fwnode_property_read_u32_array ( link ,
2019-05-01 18:57:33 +03:00
" mipi-sdw-supported-clock-gears " ,
prop - > clk_gears ,
prop - > num_clk_gears ) ;
2017-12-14 08:49:35 +03:00
}
fwnode_property_read_u32 ( link , " mipi-sdw-default-frame-rate " ,
2019-05-01 18:57:33 +03:00
& prop - > default_frame_rate ) ;
2017-12-14 08:49:35 +03:00
fwnode_property_read_u32 ( link , " mipi-sdw-default-frame-row-size " ,
2019-05-01 18:57:33 +03:00
& prop - > default_row ) ;
2017-12-14 08:49:35 +03:00
fwnode_property_read_u32 ( link , " mipi-sdw-default-frame-col-size " ,
2019-05-01 18:57:33 +03:00
& prop - > default_col ) ;
2017-12-14 08:49:35 +03:00
prop - > dynamic_frame = fwnode_property_read_bool ( link ,
" mipi-sdw-dynamic-frame-shape " ) ;
fwnode_property_read_u32 ( link , " mipi-sdw-command-error-threshold " ,
2019-05-01 18:57:33 +03:00
& prop - > err_threshold ) ;
2017-12-14 08:49:35 +03:00
return 0 ;
}
EXPORT_SYMBOL ( sdw_master_read_prop ) ;
static int sdw_slave_read_dp0 ( struct sdw_slave * slave ,
2019-05-01 18:57:33 +03:00
struct fwnode_handle * port ,
struct sdw_dp0_prop * dp0 )
2017-12-14 08:49:35 +03:00
{
int nval ;
fwnode_property_read_u32 ( port , " mipi-sdw-port-max-wordlength " ,
2019-05-01 18:57:33 +03:00
& dp0 - > max_word ) ;
2017-12-14 08:49:35 +03:00
fwnode_property_read_u32 ( port , " mipi-sdw-port-min-wordlength " ,
2019-05-01 18:57:33 +03:00
& dp0 - > min_word ) ;
2017-12-14 08:49:35 +03:00
nval = fwnode_property_read_u32_array ( port ,
" mipi-sdw-port-wordlength-configs " , NULL , 0 ) ;
if ( nval > 0 ) {
dp0 - > num_words = nval ;
dp0 - > words = devm_kcalloc ( & slave - > dev ,
2019-05-01 18:57:33 +03:00
dp0 - > num_words , sizeof ( * dp0 - > words ) ,
GFP_KERNEL ) ;
2017-12-14 08:49:35 +03:00
if ( ! dp0 - > words )
return - ENOMEM ;
fwnode_property_read_u32_array ( port ,
" mipi-sdw-port-wordlength-configs " ,
dp0 - > words , dp0 - > num_words ) ;
}
2019-05-01 18:57:33 +03:00
dp0 - > flow_controlled = fwnode_property_read_bool ( port ,
" mipi-sdw-bra-flow-controlled " ) ;
2017-12-14 08:49:35 +03:00
2019-05-01 18:57:33 +03:00
dp0 - > simple_ch_prep_sm = fwnode_property_read_bool ( port ,
" mipi-sdw-simplified-channel-prepare-sm " ) ;
2017-12-14 08:49:35 +03:00
2019-05-01 18:57:33 +03:00
dp0 - > device_interrupts = fwnode_property_read_bool ( port ,
" mipi-sdw-imp-def-dp0-interrupts-supported " ) ;
2017-12-14 08:49:35 +03:00
return 0 ;
}
static int sdw_slave_read_dpn ( struct sdw_slave * slave ,
2019-05-01 18:57:33 +03:00
struct sdw_dpn_prop * dpn , int count , int ports ,
char * type )
2017-12-14 08:49:35 +03:00
{
struct fwnode_handle * node ;
u32 bit , i = 0 ;
int nval ;
unsigned long addr ;
char name [ 40 ] ;
addr = ports ;
/* valid ports are 1 to 14 so apply mask */
addr & = GENMASK ( 14 , 1 ) ;
for_each_set_bit ( bit , & addr , 32 ) {
snprintf ( name , sizeof ( name ) ,
2019-05-01 18:57:33 +03:00
" mipi-sdw-dp-%d-%s-subproperties " , bit , type ) ;
2017-12-14 08:49:35 +03:00
dpn [ i ] . num = bit ;
node = device_get_named_child_node ( & slave - > dev , name ) ;
if ( ! node ) {
dev_err ( & slave - > dev , " %s dpN not found \n " , name ) ;
return - EIO ;
}
fwnode_property_read_u32 ( node , " mipi-sdw-port-max-wordlength " ,
2019-05-01 18:57:33 +03:00
& dpn [ i ] . max_word ) ;
2017-12-14 08:49:35 +03:00
fwnode_property_read_u32 ( node , " mipi-sdw-port-min-wordlength " ,
2019-05-01 18:57:33 +03:00
& dpn [ i ] . min_word ) ;
2017-12-14 08:49:35 +03:00
nval = fwnode_property_read_u32_array ( node ,
" mipi-sdw-port-wordlength-configs " , NULL , 0 ) ;
if ( nval > 0 ) {
dpn [ i ] . num_words = nval ;
dpn [ i ] . words = devm_kcalloc ( & slave - > dev ,
2019-05-01 18:57:33 +03:00
dpn [ i ] . num_words ,
sizeof ( * dpn [ i ] . words ) ,
GFP_KERNEL ) ;
2017-12-14 08:49:35 +03:00
if ( ! dpn [ i ] . words )
return - ENOMEM ;
fwnode_property_read_u32_array ( node ,
" mipi-sdw-port-wordlength-configs " ,
dpn [ i ] . words , dpn [ i ] . num_words ) ;
}
fwnode_property_read_u32 ( node , " mipi-sdw-data-port-type " ,
2019-05-01 18:57:33 +03:00
& dpn [ i ] . type ) ;
2017-12-14 08:49:35 +03:00
fwnode_property_read_u32 ( node ,
2019-05-01 18:57:33 +03:00
" mipi-sdw-max-grouping-supported " ,
& dpn [ i ] . max_grouping ) ;
2017-12-14 08:49:35 +03:00
dpn [ i ] . simple_ch_prep_sm = fwnode_property_read_bool ( node ,
" mipi-sdw-simplified-channelprepare-sm " ) ;
fwnode_property_read_u32 ( node ,
2019-05-01 18:57:33 +03:00
" mipi-sdw-port-channelprepare-timeout " ,
& dpn [ i ] . ch_prep_timeout ) ;
2017-12-14 08:49:35 +03:00
fwnode_property_read_u32 ( node ,
" mipi-sdw-imp-def-dpn-interrupts-supported " ,
& dpn [ i ] . device_interrupts ) ;
fwnode_property_read_u32 ( node , " mipi-sdw-min-channel-number " ,
2019-05-01 18:57:33 +03:00
& dpn [ i ] . min_ch ) ;
2017-12-14 08:49:35 +03:00
fwnode_property_read_u32 ( node , " mipi-sdw-max-channel-number " ,
2019-05-01 18:57:33 +03:00
& dpn [ i ] . max_ch ) ;
2017-12-14 08:49:35 +03:00
nval = fwnode_property_read_u32_array ( node ,
" mipi-sdw-channel-number-list " , NULL , 0 ) ;
if ( nval > 0 ) {
dpn [ i ] . num_ch = nval ;
dpn [ i ] . ch = devm_kcalloc ( & slave - > dev , dpn [ i ] . num_ch ,
2019-05-01 18:57:33 +03:00
sizeof ( * dpn [ i ] . ch ) ,
GFP_KERNEL ) ;
2017-12-14 08:49:35 +03:00
if ( ! dpn [ i ] . ch )
return - ENOMEM ;
fwnode_property_read_u32_array ( node ,
" mipi-sdw-channel-number-list " ,
dpn [ i ] . ch , dpn [ i ] . num_ch ) ;
}
nval = fwnode_property_read_u32_array ( node ,
" mipi-sdw-channel-combination-list " , NULL , 0 ) ;
if ( nval > 0 ) {
dpn [ i ] . num_ch_combinations = nval ;
dpn [ i ] . ch_combinations = devm_kcalloc ( & slave - > dev ,
dpn [ i ] . num_ch_combinations ,
sizeof ( * dpn [ i ] . ch_combinations ) ,
GFP_KERNEL ) ;
if ( ! dpn [ i ] . ch_combinations )
return - ENOMEM ;
fwnode_property_read_u32_array ( node ,
" mipi-sdw-channel-combination-list " ,
dpn [ i ] . ch_combinations ,
dpn [ i ] . num_ch_combinations ) ;
}
fwnode_property_read_u32 ( node ,
" mipi-sdw-modes-supported " , & dpn [ i ] . modes ) ;
fwnode_property_read_u32 ( node , " mipi-sdw-max-async-buffer " ,
2019-05-01 18:57:33 +03:00
& dpn [ i ] . max_async_buffer ) ;
2017-12-14 08:49:35 +03:00
dpn [ i ] . block_pack_mode = fwnode_property_read_bool ( node ,
" mipi-sdw-block-packing-mode " ) ;
fwnode_property_read_u32 ( node , " mipi-sdw-port-encoding-type " ,
2019-05-01 18:57:33 +03:00
& dpn [ i ] . port_encoding ) ;
2017-12-14 08:49:35 +03:00
/* TODO: Read audio mode */
i + + ;
}
return 0 ;
}
/**
* sdw_slave_read_prop ( ) - Read Slave properties
* @ slave : SDW Slave
*/
int sdw_slave_read_prop ( struct sdw_slave * slave )
{
struct sdw_slave_prop * prop = & slave - > prop ;
struct device * dev = & slave - > dev ;
struct fwnode_handle * port ;
int num_of_ports , nval , i , dp0 = 0 ;
device_property_read_u32 ( dev , " mipi-sdw-sw-interface-revision " ,
2019-05-01 18:57:33 +03:00
& prop - > mipi_revision ) ;
2017-12-14 08:49:35 +03:00
prop - > wake_capable = device_property_read_bool ( dev ,
" mipi-sdw-wake-up-unavailable " ) ;
prop - > wake_capable = ! prop - > wake_capable ;
prop - > test_mode_capable = device_property_read_bool ( dev ,
" mipi-sdw-test-mode-supported " ) ;
prop - > clk_stop_mode1 = false ;
if ( device_property_read_bool ( dev ,
" mipi-sdw-clock-stop-mode1-supported " ) )
prop - > clk_stop_mode1 = true ;
prop - > simple_clk_stop_capable = device_property_read_bool ( dev ,
" mipi-sdw-simplified-clockstopprepare-sm-supported " ) ;
device_property_read_u32 ( dev , " mipi-sdw-clockstopprepare-timeout " ,
2019-05-01 18:57:33 +03:00
& prop - > clk_stop_timeout ) ;
2017-12-14 08:49:35 +03:00
device_property_read_u32 ( dev , " mipi-sdw-slave-channelprepare-timeout " ,
2019-05-01 18:57:33 +03:00
& prop - > ch_prep_timeout ) ;
2017-12-14 08:49:35 +03:00
device_property_read_u32 ( dev ,
" mipi-sdw-clockstopprepare-hard-reset-behavior " ,
& prop - > reset_behave ) ;
prop - > high_PHY_capable = device_property_read_bool ( dev ,
" mipi-sdw-highPHY-capable " ) ;
prop - > paging_support = device_property_read_bool ( dev ,
" mipi-sdw-paging-support " ) ;
prop - > bank_delay_support = device_property_read_bool ( dev ,
" mipi-sdw-bank-delay-support " ) ;
device_property_read_u32 ( dev ,
" mipi-sdw-port15-read-behavior " , & prop - > p15_behave ) ;
device_property_read_u32 ( dev , " mipi-sdw-master-count " ,
2019-05-01 18:57:33 +03:00
& prop - > master_count ) ;
2017-12-14 08:49:35 +03:00
device_property_read_u32 ( dev , " mipi-sdw-source-port-list " ,
2019-05-01 18:57:33 +03:00
& prop - > source_ports ) ;
2017-12-14 08:49:35 +03:00
device_property_read_u32 ( dev , " mipi-sdw-sink-port-list " ,
2019-05-01 18:57:33 +03:00
& prop - > sink_ports ) ;
2017-12-14 08:49:35 +03:00
/* Read dp0 properties */
port = device_get_named_child_node ( dev , " mipi-sdw-dp-0-subproperties " ) ;
if ( ! port ) {
dev_dbg ( dev , " DP0 node not found!! \n " ) ;
} else {
prop - > dp0_prop = devm_kzalloc ( & slave - > dev ,
2019-05-01 18:57:33 +03:00
sizeof ( * prop - > dp0_prop ) ,
GFP_KERNEL ) ;
2017-12-14 08:49:35 +03:00
if ( ! prop - > dp0_prop )
return - ENOMEM ;
sdw_slave_read_dp0 ( slave , port , prop - > dp0_prop ) ;
dp0 = 1 ;
}
/*
* Based on each DPn port , get source and sink dpn properties .
* Also , some ports can operate as both source or sink .
*/
/* Allocate memory for set bits in port lists */
nval = hweight32 ( prop - > source_ports ) ;
prop - > src_dpn_prop = devm_kcalloc ( & slave - > dev , nval ,
2019-05-01 18:57:33 +03:00
sizeof ( * prop - > src_dpn_prop ) ,
GFP_KERNEL ) ;
2017-12-14 08:49:35 +03:00
if ( ! prop - > src_dpn_prop )
return - ENOMEM ;
/* Read dpn properties for source port(s) */
sdw_slave_read_dpn ( slave , prop - > src_dpn_prop , nval ,
2019-05-01 18:57:33 +03:00
prop - > source_ports , " source " ) ;
2017-12-14 08:49:35 +03:00
nval = hweight32 ( prop - > sink_ports ) ;
prop - > sink_dpn_prop = devm_kcalloc ( & slave - > dev , nval ,
2019-05-01 18:57:33 +03:00
sizeof ( * prop - > sink_dpn_prop ) ,
GFP_KERNEL ) ;
2017-12-14 08:49:35 +03:00
if ( ! prop - > sink_dpn_prop )
return - ENOMEM ;
/* Read dpn properties for sink port(s) */
sdw_slave_read_dpn ( slave , prop - > sink_dpn_prop , nval ,
2019-05-01 18:57:33 +03:00
prop - > sink_ports , " sink " ) ;
2017-12-14 08:49:35 +03:00
/* some ports are bidirectional so check total ports by ORing */
nval = prop - > source_ports | prop - > sink_ports ;
num_of_ports = hweight32 ( nval ) + dp0 ; /* add DP0 */
/* Allocate port_ready based on num_of_ports */
slave - > port_ready = devm_kcalloc ( & slave - > dev , num_of_ports ,
2019-05-01 18:57:33 +03:00
sizeof ( * slave - > port_ready ) ,
GFP_KERNEL ) ;
2017-12-14 08:49:35 +03:00
if ( ! slave - > port_ready )
return - ENOMEM ;
/* Initialize completion */
for ( i = 0 ; i < num_of_ports ; i + + )
init_completion ( & slave - > port_ready [ i ] ) ;
return 0 ;
}
EXPORT_SYMBOL ( sdw_slave_read_prop ) ;