123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * Copyright (c) 2022-2023, Qualcomm Innovation Center, Inc. All rights reserved.
- * Copyright (c) 2016-2021, The Linux Foundation. All rights reserved.
- */
- #include <linux/of_platform.h>
- #include <linux/soc/qcom/msm_ext_display.h>
- #include <linux/version.h>
- #if (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 19, 0))
- #include <drm/display/drm_dp_helper.h>
- #else
- #include <drm/drm_dp_helper.h>
- #endif
- #include "dp_catalog.h"
- #include "dp_audio.h"
- #include "dp_panel.h"
- #include "dp_debug.h"
- #if defined(CONFIG_SECDP)
- #include "secdp.h"
- #if defined(CONFIG_SECDP_BIGDATA)
- #include <linux/secdp_bigdata.h>
- #endif
- #if defined(CONFIG_SECDP_SWITCH)
- #include <linux/switch.h>
- static struct switch_dev switch_secdp_audio = {
- .name = "ch_hdmi_audio",
- };
- #endif
- #endif/*CONFIG_SECDP*/
- struct dp_audio_private {
- struct platform_device *ext_pdev;
- struct platform_device *pdev;
- struct dp_catalog_audio *catalog;
- struct msm_ext_disp_init_data ext_audio_data;
- struct dp_panel *panel;
- bool ack_enabled;
- atomic_t session_on;
- bool engine_on;
- u32 channels;
- struct completion hpd_comp;
- struct workqueue_struct *notify_workqueue;
- struct delayed_work notify_delayed_work;
- struct mutex ops_lock;
- struct dp_audio dp_audio;
- atomic_t acked;
- };
- static u32 dp_audio_get_header(struct dp_catalog_audio *catalog,
- enum dp_catalog_audio_sdp_type sdp,
- enum dp_catalog_audio_header_type header)
- {
- catalog->sdp_type = sdp;
- catalog->sdp_header = header;
- catalog->get_header(catalog);
- return catalog->data;
- }
- static void dp_audio_set_header(struct dp_catalog_audio *catalog,
- u32 data,
- enum dp_catalog_audio_sdp_type sdp,
- enum dp_catalog_audio_header_type header)
- {
- catalog->sdp_type = sdp;
- catalog->sdp_header = header;
- catalog->data = data;
- catalog->set_header(catalog);
- }
- static void dp_audio_stream_sdp(struct dp_audio_private *audio)
- {
- struct dp_catalog_audio *catalog = audio->catalog;
- u32 value, new_value;
- u8 parity_byte;
- /* Config header and parity byte 1 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_1);
- value &= 0x0000ffff;
- new_value = 0x02;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_1_BIT)
- | (parity_byte << PARITY_BYTE_1_BIT));
- DP_DEBUG("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_1);
- /* Config header and parity byte 2 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_2);
- value &= 0xffff0000;
- new_value = 0x0;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_2_BIT)
- | (parity_byte << PARITY_BYTE_2_BIT));
- DP_DEBUG("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_2);
- /* Config header and parity byte 3 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_3);
- value &= 0x0000ffff;
- new_value = audio->channels - 1;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_3_BIT)
- | (parity_byte << PARITY_BYTE_3_BIT));
- DP_DEBUG("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_3);
- }
- static void dp_audio_timestamp_sdp(struct dp_audio_private *audio)
- {
- struct dp_catalog_audio *catalog = audio->catalog;
- u32 value, new_value;
- u8 parity_byte;
- /* Config header and parity byte 1 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_1);
- value &= 0x0000ffff;
- new_value = 0x1;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_1_BIT)
- | (parity_byte << PARITY_BYTE_1_BIT));
- DP_DEBUG("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_1);
- /* Config header and parity byte 2 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_2);
- value &= 0xffff0000;
- new_value = 0x17;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_2_BIT)
- | (parity_byte << PARITY_BYTE_2_BIT));
- DP_DEBUG("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_2);
- /* Config header and parity byte 3 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_3);
- value &= 0x0000ffff;
- new_value = (0x0 | (0x11 << 2));
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_3_BIT)
- | (parity_byte << PARITY_BYTE_3_BIT));
- DP_DEBUG("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_3);
- }
- static void dp_audio_infoframe_sdp(struct dp_audio_private *audio)
- {
- struct dp_catalog_audio *catalog = audio->catalog;
- u32 value, new_value;
- u8 parity_byte;
- /* Config header and parity byte 1 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_1);
- value &= 0x0000ffff;
- new_value = 0x84;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_1_BIT)
- | (parity_byte << PARITY_BYTE_1_BIT));
- DP_DEBUG("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_1);
- /* Config header and parity byte 2 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_2);
- value &= 0xffff0000;
- new_value = 0x1b;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_2_BIT)
- | (parity_byte << PARITY_BYTE_2_BIT));
- DP_DEBUG("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_2);
- /* Config header and parity byte 3 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_3);
- value &= 0x0000ffff;
- new_value = (0x0 | (0x11 << 2));
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_3_BIT)
- | (parity_byte << PARITY_BYTE_3_BIT));
- DP_DEBUG("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
- new_value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_3);
- }
- static void dp_audio_copy_management_sdp(struct dp_audio_private *audio)
- {
- struct dp_catalog_audio *catalog = audio->catalog;
- u32 value, new_value;
- u8 parity_byte;
- /* Config header and parity byte 1 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_1);
- value &= 0x0000ffff;
- new_value = 0x05;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_1_BIT)
- | (parity_byte << PARITY_BYTE_1_BIT));
- DP_DEBUG("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_1);
- /* Config header and parity byte 2 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_2);
- value &= 0xffff0000;
- new_value = 0x0F;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_2_BIT)
- | (parity_byte << PARITY_BYTE_2_BIT));
- DP_DEBUG("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_2);
- /* Config header and parity byte 3 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_3);
- value &= 0x0000ffff;
- new_value = 0x0;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_3_BIT)
- | (parity_byte << PARITY_BYTE_3_BIT));
- DP_DEBUG("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_3);
- }
- static void dp_audio_isrc_sdp(struct dp_audio_private *audio)
- {
- struct dp_catalog_audio *catalog = audio->catalog;
- u32 value, new_value;
- u8 parity_byte;
- /* Config header and parity byte 1 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_1);
- value &= 0x0000ffff;
- new_value = 0x06;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_1_BIT)
- | (parity_byte << PARITY_BYTE_1_BIT));
- DP_DEBUG("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_1);
- /* Config header and parity byte 2 */
- value = dp_audio_get_header(catalog,
- DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_2);
- value &= 0xffff0000;
- new_value = 0x0F;
- parity_byte = dp_header_get_parity(new_value);
- value |= ((new_value << HEADER_BYTE_2_BIT)
- | (parity_byte << PARITY_BYTE_2_BIT));
- DP_DEBUG("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
- value, parity_byte);
- dp_audio_set_header(catalog, value,
- DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_2);
- }
- static void dp_audio_setup_sdp(struct dp_audio_private *audio)
- {
- if (!atomic_read(&audio->session_on)) {
- DP_WARN("session inactive\n");
- return;
- }
- /* always program stream 0 first before actual stream cfg */
- audio->catalog->stream_id = DP_STREAM_0;
- audio->catalog->config_sdp(audio->catalog);
- if (audio->panel->stream_id == DP_STREAM_1) {
- audio->catalog->stream_id = DP_STREAM_1;
- audio->catalog->config_sdp(audio->catalog);
- }
- dp_audio_stream_sdp(audio);
- dp_audio_timestamp_sdp(audio);
- dp_audio_infoframe_sdp(audio);
- dp_audio_copy_management_sdp(audio);
- dp_audio_isrc_sdp(audio);
- }
- static void dp_audio_setup_acr(struct dp_audio_private *audio)
- {
- u32 select = 0;
- struct dp_catalog_audio *catalog = audio->catalog;
- if (!atomic_read(&audio->session_on)) {
- DP_WARN("session inactive\n");
- return;
- }
- switch (audio->dp_audio.bw_code) {
- case DP_LINK_BW_1_62:
- select = 0;
- break;
- case DP_LINK_BW_2_7:
- select = 1;
- break;
- case DP_LINK_BW_5_4:
- select = 2;
- break;
- case DP_LINK_BW_8_1:
- select = 3;
- break;
- default:
- DP_DEBUG("Unknown link rate\n");
- select = 0;
- break;
- }
- catalog->data = select;
- catalog->config_acr(catalog);
- }
- static void dp_audio_enable(struct dp_audio_private *audio, bool enable)
- {
- struct dp_catalog_audio *catalog = audio->catalog;
- audio->engine_on = enable;
- if (!atomic_read(&audio->session_on)) {
- DP_WARN("session inactive. enable=%d\n", enable);
- return;
- }
- catalog->data = enable;
- if (audio->panel->get_panel_on(audio->panel))
- catalog->enable(catalog);
- }
- static struct dp_audio_private *dp_audio_get_data(struct platform_device *pdev)
- {
- struct msm_ext_disp_data *ext_data;
- struct dp_audio *dp_audio;
- if (!pdev) {
- DP_ERR("invalid input\n");
- return ERR_PTR(-ENODEV);
- }
- ext_data = platform_get_drvdata(pdev);
- if (!ext_data) {
- DP_ERR("invalid ext disp data\n");
- return ERR_PTR(-EINVAL);
- }
- dp_audio = ext_data->intf_data;
- if (!dp_audio) {
- DP_ERR("invalid intf data\n");
- return ERR_PTR(-EINVAL);
- }
- return container_of(dp_audio, struct dp_audio_private, dp_audio);
- }
- static int dp_audio_info_setup(struct platform_device *pdev,
- struct msm_ext_disp_audio_setup_params *params)
- {
- int rc = 0;
- struct dp_audio_private *audio;
- audio = dp_audio_get_data(pdev);
- if (IS_ERR(audio)) {
- rc = PTR_ERR(audio);
- return rc;
- }
- if (audio->dp_audio.tui_active) {
- DP_DEBUG("TUI session active\n");
- return 0;
- }
- #if defined(CONFIG_SECDP_BIGDATA)
- secdp_bigdata_save_item(BD_AUD_CH, params->num_of_channels);
- secdp_bigdata_save_item(BD_AUD_FREQ, params->sample_rate_hz);
- #endif
- mutex_lock(&audio->ops_lock);
- audio->channels = params->num_of_channels;
- if (audio->panel->stream_id >= DP_STREAM_MAX) {
- DP_ERR("invalid stream id: %d\n",
- audio->panel->stream_id);
- rc = -EINVAL;
- mutex_unlock(&audio->ops_lock);
- return rc;
- }
- dp_audio_setup_sdp(audio);
- dp_audio_setup_acr(audio);
- dp_audio_enable(audio, true);
- mutex_unlock(&audio->ops_lock);
- DP_DEBUG("audio stream configured\n");
- return rc;
- }
- #if defined(CONFIG_SECDP)
- static void secdp_audio_use_one_sampling_freq(u8 *adb, int size)
- {
- const int one_adb_size = 3;
- int adb_count;
- if (size <= 0)
- return;
- adb_count = size / one_adb_size;
- while (adb_count > 0) {
- if (adb[1] & BIT(2))
- adb[1] = BIT(2); /* 48kHz */
- else if (adb[1] & BIT(1))
- adb[1] = BIT(1); /* 44.1kHz */
- else if (adb[1] & BIT(0))
- adb[1] = BIT(0); /* 32kHz */
- else if (adb[1] & BIT(3))
- adb[1] = BIT(3); /* 88kHz */
- else if (adb[1] & BIT(4))
- adb[1] = BIT(4); /* 96kHz */
- else if (adb[1] & BIT(5))
- adb[1] = BIT(5); /* 176kHz */
- else if (adb[1] & BIT(6))
- adb[1] = BIT(6); /* 192kHz */
- adb += one_adb_size;
- adb_count--;
- }
- }
- #endif
- static int dp_audio_get_edid_blk(struct platform_device *pdev,
- struct msm_ext_disp_audio_edid_blk *blk)
- {
- int rc = 0;
- struct dp_audio_private *audio;
- struct sde_edid_ctrl *edid;
- if (!blk) {
- DP_ERR("invalid input\n");
- return -EINVAL;
- }
- audio = dp_audio_get_data(pdev);
- if (IS_ERR(audio)) {
- rc = PTR_ERR(audio);
- goto end;
- }
- if (!audio->panel || !audio->panel->edid_ctrl) {
- DP_ERR("invalid panel data\n");
- rc = -EINVAL;
- goto end;
- }
- edid = audio->panel->edid_ctrl;
- #if defined(CONFIG_SECDP)
- if (secdp_adapter_is_legacy())
- secdp_audio_use_one_sampling_freq(edid->audio_data_block, edid->adb_size);
- #endif
- blk->audio_data_blk = edid->audio_data_block;
- blk->audio_data_blk_size = edid->adb_size;
- #if defined(CONFIG_SECDP)
- print_hex_dump(KERN_DEBUG, "AUDIO_BLK: ",
- DUMP_PREFIX_NONE, 16, 1, blk->audio_data_blk,
- blk->audio_data_blk_size, false);
- secdp_logger_hex_dump(blk->audio_data_blk, "AUDIO_BLK:",
- blk->audio_data_blk_size);
- #endif
- blk->spk_alloc_data_blk = edid->spkr_alloc_data_block;
- blk->spk_alloc_data_blk_size = edid->sadb_size;
- end:
- return rc;
- }
- static int dp_audio_get_cable_status(struct platform_device *pdev, u32 vote)
- {
- int rc = 0;
- struct dp_audio_private *audio;
- audio = dp_audio_get_data(pdev);
- if (IS_ERR(audio)) {
- rc = PTR_ERR(audio);
- goto end;
- }
- return atomic_read(&audio->session_on);
- end:
- return rc;
- }
- static int dp_audio_get_intf_id(struct platform_device *pdev)
- {
- int rc = 0;
- struct dp_audio_private *audio;
- audio = dp_audio_get_data(pdev);
- if (IS_ERR(audio)) {
- rc = PTR_ERR(audio);
- goto end;
- }
- return EXT_DISPLAY_TYPE_DP;
- end:
- return rc;
- }
- static void dp_audio_teardown_done(struct platform_device *pdev)
- {
- struct dp_audio_private *audio;
- audio = dp_audio_get_data(pdev);
- if (IS_ERR(audio))
- return;
- if (audio->dp_audio.tui_active) {
- DP_DEBUG("TUI session active\n");
- return;
- }
- if (audio->panel->stream_id >= DP_STREAM_MAX) {
- DP_WARN("invalid stream id: %d\n",
- audio->panel->stream_id);
- return;
- }
- mutex_lock(&audio->ops_lock);
- dp_audio_enable(audio, false);
- mutex_unlock(&audio->ops_lock);
- atomic_set(&audio->acked, 1);
- complete_all(&audio->hpd_comp);
- DP_DEBUG("audio engine disabled\n");
- }
- static int dp_audio_ack_done(struct platform_device *pdev, u32 ack)
- {
- int rc = 0, ack_hpd;
- struct dp_audio_private *audio;
- audio = dp_audio_get_data(pdev);
- if (IS_ERR(audio)) {
- rc = PTR_ERR(audio);
- goto end;
- }
- if (ack & AUDIO_ACK_SET_ENABLE) {
- audio->ack_enabled = ack & AUDIO_ACK_ENABLE ?
- true : false;
- DP_DEBUG("audio ack feature %s\n",
- audio->ack_enabled ? "enabled" : "disabled");
- goto end;
- }
- if (!audio->ack_enabled)
- goto end;
- ack_hpd = ack & AUDIO_ACK_CONNECT;
- DP_DEBUG("acknowledging audio (%d)\n", ack_hpd);
- if (!audio->engine_on) {
- atomic_set(&audio->acked, 1);
- complete_all(&audio->hpd_comp);
- }
- end:
- return rc;
- }
- static int dp_audio_codec_ready(struct platform_device *pdev)
- {
- int rc = 0;
- struct dp_audio_private *audio;
- audio = dp_audio_get_data(pdev);
- if (IS_ERR(audio)) {
- DP_ERR("invalid input\n");
- rc = PTR_ERR(audio);
- goto end;
- }
- queue_delayed_work(audio->notify_workqueue,
- &audio->notify_delayed_work, HZ/4);
- end:
- return rc;
- }
- static int dp_audio_register_ext_disp(struct dp_audio_private *audio)
- {
- int rc = 0;
- struct device_node *pd = NULL;
- const char *phandle = "qcom,ext-disp";
- struct msm_ext_disp_init_data *ext;
- struct msm_ext_disp_audio_codec_ops *ops;
- ext = &audio->ext_audio_data;
- ops = &ext->codec_ops;
- ext->codec.type = EXT_DISPLAY_TYPE_DP;
- ext->codec.ctrl_id = 0;
- ext->codec.stream_id = audio->panel->stream_id;
- ext->pdev = audio->pdev;
- ext->intf_data = &audio->dp_audio;
- ops->audio_info_setup = dp_audio_info_setup;
- ops->get_audio_edid_blk = dp_audio_get_edid_blk;
- ops->cable_status = dp_audio_get_cable_status;
- ops->get_intf_id = dp_audio_get_intf_id;
- ops->teardown_done = dp_audio_teardown_done;
- ops->acknowledge = dp_audio_ack_done;
- ops->ready = dp_audio_codec_ready;
- if (!audio->pdev->dev.of_node) {
- DP_ERR("cannot find audio dev.of_node\n");
- rc = -ENODEV;
- goto end;
- }
- pd = of_parse_phandle(audio->pdev->dev.of_node, phandle, 0);
- if (!pd) {
- DP_ERR("cannot parse %s handle\n", phandle);
- rc = -ENODEV;
- goto end;
- }
- audio->ext_pdev = of_find_device_by_node(pd);
- if (!audio->ext_pdev) {
- DP_ERR("cannot find %s pdev\n", phandle);
- rc = -ENODEV;
- goto end;
- }
- #if IS_ENABLED(CONFIG_MSM_EXT_DISPLAY)
- rc = msm_ext_disp_register_intf(audio->ext_pdev, ext);
- if (rc)
- DP_ERR("failed to register disp\n");
- #endif
- end:
- if (pd)
- of_node_put(pd);
- return rc;
- }
- static int dp_audio_deregister_ext_disp(struct dp_audio_private *audio)
- {
- int rc = 0;
- struct device_node *pd = NULL;
- const char *phandle = "qcom,ext-disp";
- struct msm_ext_disp_init_data *ext;
- ext = &audio->ext_audio_data;
- if (!audio->pdev->dev.of_node) {
- DP_ERR("cannot find audio dev.of_node\n");
- rc = -ENODEV;
- goto end;
- }
- pd = of_parse_phandle(audio->pdev->dev.of_node, phandle, 0);
- if (!pd) {
- DP_ERR("cannot parse %s handle\n", phandle);
- rc = -ENODEV;
- goto end;
- }
- audio->ext_pdev = of_find_device_by_node(pd);
- if (!audio->ext_pdev) {
- DP_ERR("cannot find %s pdev\n", phandle);
- rc = -ENODEV;
- goto end;
- }
- #if IS_ENABLED(CONFIG_MSM_EXT_DISPLAY)
- rc = msm_ext_disp_deregister_intf(audio->ext_pdev, ext);
- if (rc)
- DP_ERR("failed to deregister disp\n");
- #endif
- end:
- return rc;
- }
- #if defined(CONFIG_SECDP_SWITCH)
- extern int secdp_get_audio_ch(void);
- #endif
- static int dp_audio_notify(struct dp_audio_private *audio, u32 state)
- {
- int rc = 0;
- struct msm_ext_disp_init_data *ext = &audio->ext_audio_data;
- atomic_set(&audio->acked, 0);
- if (!ext->intf_ops.audio_notify) {
- DP_ERR("audio notify not defined\n");
- goto end;
- }
- reinit_completion(&audio->hpd_comp);
- rc = ext->intf_ops.audio_notify(audio->ext_pdev,
- &ext->codec, state);
- if (rc)
- goto end;
- #if defined(CONFIG_SECDP_SWITCH)
- {
- if (!audio->dp_audio.has_mst) {
- int audio_ch = state ? secdp_get_audio_ch() : -1;
- switch_set_state(&switch_secdp_audio, audio_ch);
- DP_INFO("secdp audio state:0x%x\n", audio_ch);
- }
- }
- #endif
- if (atomic_read(&audio->acked))
- goto end;
- if (state == EXT_DISPLAY_CABLE_DISCONNECT && !audio->engine_on)
- goto end;
- if (state == EXT_DISPLAY_CABLE_CONNECT)
- goto end;
- rc = wait_for_completion_timeout(&audio->hpd_comp, HZ * 4);
- if (!rc) {
- DP_ERR("timeout. state=%d err=%d\n", state, rc);
- rc = -ETIMEDOUT;
- goto end;
- }
- DP_DEBUG("success\n");
- end:
- return rc;
- }
- static int dp_audio_config(struct dp_audio_private *audio, u32 state)
- {
- int rc = 0;
- struct msm_ext_disp_init_data *ext = &audio->ext_audio_data;
- if (!ext || !ext->intf_ops.audio_config) {
- DP_ERR("audio_config not defined\n");
- goto end;
- }
- /*
- * DP Audio sets default STREAM_0 only, other streams are
- * set by audio driver based on the hardware/software support.
- */
- if (audio->panel->stream_id == DP_STREAM_0) {
- rc = ext->intf_ops.audio_config(audio->ext_pdev,
- &ext->codec, state);
- if (rc)
- DP_ERR("failed to config audio, err=%d\n",
- rc);
- }
- end:
- return rc;
- }
- static int dp_audio_on(struct dp_audio *dp_audio)
- {
- int rc = 0;
- struct dp_audio_private *audio;
- struct msm_ext_disp_init_data *ext;
- if (!dp_audio) {
- DP_ERR("invalid input\n");
- return -EINVAL;
- }
- #if defined(CONFIG_SECDP)
- if (!secdp_get_cable_status()) {
- DP_INFO("cable is out\n");
- return -EINVAL;
- }
- #endif
- audio = container_of(dp_audio, struct dp_audio_private, dp_audio);
- if (IS_ERR(audio)) {
- DP_ERR("invalid input\n");
- return -EINVAL;
- }
- dp_audio_register_ext_disp(audio);
- ext = &audio->ext_audio_data;
- atomic_set(&audio->session_on, 1);
- rc = dp_audio_config(audio, EXT_DISPLAY_CABLE_CONNECT);
- if (rc)
- goto end;
- rc = dp_audio_notify(audio, EXT_DISPLAY_CABLE_CONNECT);
- if (rc)
- goto end;
- DP_DEBUG("success\n");
- end:
- return rc;
- }
- static int dp_audio_off(struct dp_audio *dp_audio, bool skip_wait)
- {
- int rc = 0;
- struct dp_audio_private *audio;
- struct msm_ext_disp_init_data *ext;
- bool work_pending = false;
- if (!dp_audio) {
- DP_ERR("invalid input\n");
- return -EINVAL;
- }
- audio = container_of(dp_audio, struct dp_audio_private, dp_audio);
- if (!atomic_read(&audio->session_on)) {
- DP_DEBUG("audio already off\n");
- return rc;
- }
- ext = &audio->ext_audio_data;
- #if defined(CONFIG_SECDP)
- if (!atomic_read(&audio->session_on)) {
- DP_INFO("dp audio already off\n");
- return rc;
- }
- #endif
- work_pending = cancel_delayed_work_sync(&audio->notify_delayed_work);
- if (work_pending)
- DP_DEBUG("pending notification work completed\n");
- if (!skip_wait) {
- rc = dp_audio_notify(audio, EXT_DISPLAY_CABLE_DISCONNECT);
- if (rc)
- goto end;
- }
- DP_DEBUG("success\n");
- end:
- dp_audio_config(audio, EXT_DISPLAY_CABLE_DISCONNECT);
- atomic_set(&audio->session_on, 0);
- audio->engine_on = false;
- dp_audio_deregister_ext_disp(audio);
- return rc;
- }
- static void dp_audio_notify_work_fn(struct work_struct *work)
- {
- struct dp_audio_private *audio;
- struct delayed_work *dw = to_delayed_work(work);
- audio = container_of(dw, struct dp_audio_private, notify_delayed_work);
- dp_audio_notify(audio, EXT_DISPLAY_CABLE_CONNECT);
- }
- #if defined(CONFIG_SECDP_SWITCH)
- int secdp_audio_register_switch(struct dp_audio *dp_audio)
- {
- int rc = 0;
- if (dp_audio->has_mst) {
- DP_INFO("skip switch register\n");
- rc = -ENODEV;
- goto end;
- }
- rc = switch_dev_register(&switch_secdp_audio);
- if (rc) {
- DP_INFO("Failed to register secdp_audio switch %d\n", rc);
- rc = -ENODEV;
- goto end;
- }
- DP_INFO("secdp_audio register success\n");
- end:
- return rc;
- }
- static void secdp_audio_unregister_switch(struct dp_audio_private *audio)
- {
- struct dp_audio *dp_audio = &audio->dp_audio;
- if (dp_audio->has_mst) {
- DP_DEBUG("skip switch unregister\n");
- return;
- }
- switch_dev_unregister(&switch_secdp_audio);
- DP_INFO("secdp_audio unregister success\n");
- }
- #endif
- static int dp_audio_create_notify_workqueue(struct dp_audio_private *audio)
- {
- audio->notify_workqueue = create_workqueue("sdm_dp_audio_notify");
- if (IS_ERR_OR_NULL(audio->notify_workqueue)) {
- DP_ERR("Error creating notify_workqueue\n");
- return -EPERM;
- }
- INIT_DELAYED_WORK(&audio->notify_delayed_work, dp_audio_notify_work_fn);
- return 0;
- }
- static void dp_audio_destroy_notify_workqueue(struct dp_audio_private *audio)
- {
- if (audio->notify_workqueue)
- destroy_workqueue(audio->notify_workqueue);
- #if defined(CONFIG_SECDP_SWITCH)
- secdp_audio_unregister_switch(audio);
- #endif
- }
- struct dp_audio *dp_audio_get(struct platform_device *pdev,
- struct dp_panel *panel,
- struct dp_catalog_audio *catalog)
- {
- int rc = 0;
- struct dp_audio_private *audio;
- struct dp_audio *dp_audio;
- if (!pdev || !panel || !catalog) {
- DP_ERR("invalid input\n");
- rc = -EINVAL;
- goto error;
- }
- audio = devm_kzalloc(&pdev->dev, sizeof(*audio), GFP_KERNEL);
- if (!audio) {
- rc = -ENOMEM;
- goto error;
- }
- rc = dp_audio_create_notify_workqueue(audio);
- if (rc)
- goto error_notify_workqueue;
- init_completion(&audio->hpd_comp);
- audio->pdev = pdev;
- audio->panel = panel;
- audio->catalog = catalog;
- atomic_set(&audio->acked, 0);
- dp_audio = &audio->dp_audio;
- mutex_init(&audio->ops_lock);
- dp_audio->on = dp_audio_on;
- dp_audio->off = dp_audio_off;
- catalog->init(catalog);
- return dp_audio;
- error_notify_workqueue:
- devm_kfree(&pdev->dev, audio);
- error:
- return ERR_PTR(rc);
- }
- void dp_audio_put(struct dp_audio *dp_audio)
- {
- struct dp_audio_private *audio;
- if (!dp_audio)
- return;
- audio = container_of(dp_audio, struct dp_audio_private, dp_audio);
- mutex_destroy(&audio->ops_lock);
- dp_audio_destroy_notify_workqueue(audio);
- devm_kfree(&audio->pdev->dev, audio);
- }
|