2019-04-12 11:08:56 -05:00
// SPDX-License-Identifier: (GPL-2.0 OR BSD-3-Clause)
//
// This file is provided under a dual BSD/GPLv2 license. When using or
// redistributing this file, you may do so under either license.
//
// Copyright(c) 2018 Intel Corporation. All rights reserved.
//
// Authors: Keyon Jie <yang.jie@linux.intel.com>
//
# include <linux/module.h>
# include <sound/hdaudio_ext.h>
2019-07-22 09:13:53 -05:00
# include <sound/hda_register.h>
2019-04-12 11:08:56 -05:00
# include <sound/hda_codec.h>
# include <sound/hda_i915.h>
# include <sound/sof.h>
# include "../ops.h"
# include "hda.h"
# if IS_ENABLED(CONFIG_SND_SOC_SOF_HDA_AUDIO_CODEC)
# include "../../codecs/hdac_hda.h"
# endif /* CONFIG_SND_SOC_SOF_HDA_AUDIO_CODEC */
# if IS_ENABLED(CONFIG_SND_SOC_SOF_HDA_AUDIO_CODEC)
# define IDISP_VID_INTEL 0x80860000
/* load the legacy HDA codec driver */
2020-01-10 17:57:50 -06:00
static int hda_codec_load_module ( struct hda_codec * codec )
2019-04-12 11:08:56 -05:00
{
2020-01-10 17:57:50 -06:00
# ifdef MODULE
2019-04-12 11:08:56 -05:00
char alias [ MODULE_NAME_LEN ] ;
const char * module = alias ;
snd_hdac_codec_modalias ( & codec - > core , alias , sizeof ( alias ) ) ;
dev_dbg ( & codec - > core . dev , " loading codec module: %s \n " , module ) ;
request_module ( module ) ;
# endif
2020-01-10 17:57:50 -06:00
return device_attach ( hda_codec_dev ( codec ) ) ;
}
2019-04-12 11:08:56 -05:00
2019-07-22 09:13:53 -05:00
/* enable controller wake up event for all codecs with jack connectors */
void hda_codec_jack_wake_enable ( struct snd_sof_dev * sdev )
{
struct hda_bus * hbus = sof_to_hbus ( sdev ) ;
struct hdac_bus * bus = sof_to_bus ( sdev ) ;
struct hda_codec * codec ;
unsigned int mask = 0 ;
list_for_each_codec ( codec , hbus )
if ( codec - > jacktbl . used )
mask | = BIT ( codec - > core . addr ) ;
snd_hdac_chip_updatew ( bus , WAKEEN , STATESTS_INT_MASK , mask ) ;
}
/* check jack status after resuming from suspend mode */
void hda_codec_jack_check ( struct snd_sof_dev * sdev )
{
struct hda_bus * hbus = sof_to_hbus ( sdev ) ;
struct hdac_bus * bus = sof_to_bus ( sdev ) ;
struct hda_codec * codec ;
/* disable controller Wake Up event*/
snd_hdac_chip_updatew ( bus , WAKEEN , STATESTS_INT_MASK , 0 ) ;
list_for_each_codec ( codec , hbus )
/*
* Wake up all jack - detecting codecs regardless whether an event
* has been recorded in STATESTS
*/
if ( codec - > jacktbl . used )
schedule_delayed_work ( & codec - > jackpoll_work ,
codec - > jackpoll_interval ) ;
}
# else
void hda_codec_jack_wake_enable ( struct snd_sof_dev * sdev ) { }
void hda_codec_jack_check ( struct snd_sof_dev * sdev ) { }
2019-04-12 11:08:56 -05:00
# endif /* CONFIG_SND_SOC_SOF_HDA_AUDIO_CODEC */
2019-12-17 14:22:29 -06:00
EXPORT_SYMBOL_NS ( hda_codec_jack_wake_enable , SND_SOC_SOF_HDA_AUDIO_CODEC ) ;
EXPORT_SYMBOL_NS ( hda_codec_jack_check , SND_SOC_SOF_HDA_AUDIO_CODEC ) ;
2019-04-12 11:08:56 -05:00
/* probe individual codec */
2019-12-04 15:15:52 -06:00
static int hda_codec_probe ( struct snd_sof_dev * sdev , int address ,
bool hda_codec_use_common_hdmi )
2019-04-12 11:08:56 -05:00
{
# if IS_ENABLED(CONFIG_SND_SOC_SOF_HDA_AUDIO_CODEC)
struct hdac_hda_priv * hda_priv ;
# endif
2019-07-22 09:13:53 -05:00
struct hda_bus * hbus = sof_to_hbus ( sdev ) ;
struct hdac_device * hdev ;
2019-04-12 11:08:56 -05:00
u32 hda_cmd = ( address < < 28 ) | ( AC_NODE_ROOT < < 20 ) |
( AC_VERB_PARAMETERS < < 8 ) | AC_PAR_VENDOR_ID ;
u32 resp = - 1 ;
int ret ;
mutex_lock ( & hbus - > core . cmd_mutex ) ;
snd_hdac_bus_send_cmd ( & hbus - > core , hda_cmd ) ;
snd_hdac_bus_get_response ( & hbus - > core , address , & resp ) ;
mutex_unlock ( & hbus - > core . cmd_mutex ) ;
if ( resp = = - 1 )
return - EIO ;
dev_dbg ( sdev - > dev , " HDA codec #%d probed OK: response: %x \n " ,
address , resp ) ;
# if IS_ENABLED(CONFIG_SND_SOC_SOF_HDA_AUDIO_CODEC)
2019-06-26 00:04:50 -07:00
hda_priv = devm_kzalloc ( sdev - > dev , sizeof ( * hda_priv ) , GFP_KERNEL ) ;
2019-04-12 11:08:56 -05:00
if ( ! hda_priv )
return - ENOMEM ;
hda_priv - > codec . bus = hbus ;
hdev = & hda_priv - > codec . core ;
ret = snd_hdac_ext_bus_device_init ( & hbus - > core , address , hdev ) ;
if ( ret < 0 )
return ret ;
2019-10-29 15:40:13 +02:00
if ( ( resp & 0xFFFF0000 ) = = IDISP_VID_INTEL )
hda_priv - > need_display_power = true ;
/*
* if common HDMI codec driver is not used , codec load
* is skipped here and hdac_hdmi is used instead
*/
2019-12-04 15:15:52 -06:00
if ( hda_codec_use_common_hdmi | |
2019-10-29 15:40:13 +02:00
( resp & 0xFFFF0000 ) ! = IDISP_VID_INTEL ) {
2019-04-12 11:08:56 -05:00
hdev - > type = HDA_DEV_LEGACY ;
2020-01-10 17:57:50 -06:00
ret = hda_codec_load_module ( & hda_priv - > codec ) ;
/*
* handle ret = = 0 ( no driver bound ) as an error , but pass
* other return codes without modification
*/
if ( ret = = 0 )
ret = - ENOENT ;
2019-04-12 11:08:56 -05:00
}
2020-01-10 17:57:50 -06:00
return ret ;
2019-04-12 11:08:56 -05:00
# else
2019-06-26 00:04:50 -07:00
hdev = devm_kzalloc ( sdev - > dev , sizeof ( * hdev ) , GFP_KERNEL ) ;
2019-04-12 11:08:56 -05:00
if ( ! hdev )
return - ENOMEM ;
ret = snd_hdac_ext_bus_device_init ( & hbus - > core , address , hdev ) ;
return ret ;
# endif
}
/* Codec initialization */
2019-12-17 18:26:13 -06:00
void hda_codec_probe_bus ( struct snd_sof_dev * sdev ,
bool hda_codec_use_common_hdmi )
2019-04-12 11:08:56 -05:00
{
struct hdac_bus * bus = sof_to_bus ( sdev ) ;
int i , ret ;
/* probe codecs in avail slots */
for ( i = 0 ; i < HDA_MAX_CODECS ; i + + ) {
if ( ! ( bus - > codec_mask & ( 1 < < i ) ) )
continue ;
2019-12-04 15:15:52 -06:00
ret = hda_codec_probe ( sdev , i , hda_codec_use_common_hdmi ) ;
2019-04-12 11:08:56 -05:00
if ( ret < 0 ) {
2019-12-17 18:26:13 -06:00
dev_warn ( bus - > dev , " codec #%d probe error, ret: %d \n " ,
i , ret ) ;
bus - > codec_mask & = ~ BIT ( i ) ;
2019-04-12 11:08:56 -05:00
}
}
}
2019-12-17 14:22:29 -06:00
EXPORT_SYMBOL_NS ( hda_codec_probe_bus , SND_SOC_SOF_HDA_AUDIO_CODEC ) ;
2019-04-12 11:08:56 -05:00
2019-10-29 15:40:13 +02:00
# if IS_ENABLED(CONFIG_SND_HDA_CODEC_HDMI) || \
IS_ENABLED ( CONFIG_SND_SOC_HDAC_HDMI )
2019-04-12 11:08:56 -05:00
2020-01-20 18:01:15 +02:00
void hda_codec_i915_display_power ( struct snd_sof_dev * sdev , bool enable )
2019-04-12 11:08:56 -05:00
{
struct hdac_bus * bus = sof_to_bus ( sdev ) ;
2020-02-06 22:02:22 +02:00
if ( HDA_IDISP_CODEC ( bus - > codec_mask ) ) {
dev_dbg ( bus - > dev , " Turning i915 HDAC power %d \n " , enable ) ;
snd_hdac_display_power ( bus , HDA_CODEC_IDX_CONTROLLER , enable ) ;
}
2019-04-12 11:08:56 -05:00
}
2020-01-20 18:01:15 +02:00
EXPORT_SYMBOL_NS ( hda_codec_i915_display_power , SND_SOC_SOF_HDA_AUDIO_CODEC_I915 ) ;
2019-04-12 11:08:56 -05:00
int hda_codec_i915_init ( struct snd_sof_dev * sdev )
{
struct hdac_bus * bus = sof_to_bus ( sdev ) ;
int ret ;
/* i915 exposes a HDA codec for HDMI audio */
ret = snd_hdac_i915_init ( bus ) ;
if ( ret < 0 )
return ret ;
2020-02-06 22:02:22 +02:00
/* codec_mask not yet known, power up for probe */
snd_hdac_display_power ( bus , HDA_CODEC_IDX_CONTROLLER , true ) ;
2019-04-12 11:08:56 -05:00
return 0 ;
}
2019-12-17 14:22:29 -06:00
EXPORT_SYMBOL_NS ( hda_codec_i915_init , SND_SOC_SOF_HDA_AUDIO_CODEC_I915 ) ;
2019-04-12 11:08:56 -05:00
int hda_codec_i915_exit ( struct snd_sof_dev * sdev )
{
struct hdac_bus * bus = sof_to_bus ( sdev ) ;
int ret ;
2020-02-06 22:02:22 +02:00
/* power down unconditionally */
snd_hdac_display_power ( bus , HDA_CODEC_IDX_CONTROLLER , false ) ;
2019-04-12 11:08:56 -05:00
ret = snd_hdac_i915_exit ( bus ) ;
return ret ;
}
2019-12-17 14:22:29 -06:00
EXPORT_SYMBOL_NS ( hda_codec_i915_exit , SND_SOC_SOF_HDA_AUDIO_CODEC_I915 ) ;
2019-04-12 11:08:56 -05:00
2019-10-29 15:40:13 +02:00
# endif
2019-04-12 11:08:56 -05:00
MODULE_LICENSE ( " Dual BSD/GPL " ) ;