From nobody Sat Apr 11 18:37:59 2026 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id 9291CC00140 for ; Mon, 8 Aug 2022 22:43:07 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S244519AbiHHWnE (ORCPT ); Mon, 8 Aug 2022 18:43:04 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:36290 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S244526AbiHHWnC (ORCPT ); Mon, 8 Aug 2022 18:43:02 -0400 Received: from hutie.ust.cz (hutie.ust.cz [185.8.165.127]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id B3285167E8 for ; Mon, 8 Aug 2022 15:43:00 -0700 (PDT) From: =?UTF-8?q?Martin=20Povi=C5=A1er?= DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=cutebit.org; s=mail; t=1659998578; bh=aQrCd+ijZmlO1Kgsm6C6rwa65AVL97zN0C3e6zixLi4=; h=From:To:Cc:Subject:Date:In-Reply-To:References; b=DiIM2ZSDHO+5xRGl9iDX/udWRADomI2tu+BybuCkpI19KgpSGU89WsAHxovyXE0Ah g+ErzsGRNeNXpTMl1lmrKQso5rqM8P0AzrKMDWTXjucDNpQ3aYJGEOGUUW2j+r85Gr fzgKJ/0j82AmKlTVa5mrjzO13NgdfuWy2cxdpivY= To: =?UTF-8?q?Martin=20Povi=C5=A1er?= , Liam Girdwood , Mark Brown , Rob Herring , Krzysztof Kozlowski , Jaroslav Kysela , Takashi Iwai , Philipp Zabel Cc: asahi@lists.linux.dev, alsa-devel@alsa-project.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org Subject: [PATCH 3/3] ASoC: apple: mca: Add locks on foreign cluster access Date: Tue, 9 Aug 2022 00:41:53 +0200 Message-Id: <20220808224153.3634-4-povik+lin@cutebit.org> In-Reply-To: <20220808224153.3634-1-povik+lin@cutebit.org> References: <20220808224153.3634-1-povik+lin@cutebit.org> MIME-Version: 1.0 Content-Type: text/plain; charset="utf-8" Content-Transfer-Encoding: quoted-printable Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org In DAI ops, accesses to the native cluster (of the DAI), and to data of clusters related to it by a DPCM frontend-backend link, should have been synchronized by the 'pcm_mutex' lock at ASoC level. What is not covered are the 'port_driver' accesses on foreign clusters to which the current cluster has no a priori relation, so fill in locking for that. (This should only matter in bizarre configurations of sharing one MCA peripheral between ASoC cards.) Signed-off-by: Martin Povi=C5=A1er --- sound/soc/apple/mca.c | 27 ++++++++++++++++++++++++--- 1 file changed, 24 insertions(+), 3 deletions(-) diff --git a/sound/soc/apple/mca.c b/sound/soc/apple/mca.c index ab41fd1a2444..1e2464e89d1c 100644 --- a/sound/soc/apple/mca.c +++ b/sound/soc/apple/mca.c @@ -159,6 +159,9 @@ struct mca_data { struct reset_control *rstc; struct device_link *pd_link; =20 + /* Mutex for accessing port_driver of foreign clusters */ + struct mutex port_mutex; + int nclusters; struct mca_cluster clusters[]; }; @@ -297,16 +300,21 @@ static bool mca_fe_clocks_in_use(struct mca_cluster *= cl) struct mca_cluster *be_cl; int stream, i; =20 + mutex_lock(&mca->port_mutex); for (i =3D 0; i < mca->nclusters; i++) { be_cl =3D &mca->clusters[i]; =20 if (be_cl->port_driver !=3D cl->no) continue; =20 - for_each_pcm_streams(stream) - if (be_cl->clocks_in_use[stream]) + for_each_pcm_streams(stream) { + if (be_cl->clocks_in_use[stream]) { + mutex_unlock(&mca->port_mutex); return true; + } + } } + mutex_unlock(&mca->port_mutex); return false; } =20 @@ -331,8 +339,10 @@ static int mca_be_prepare(struct snd_pcm_substream *su= bstream, */ if (!mca_fe_clocks_in_use(fe_cl)) { ret =3D mca_fe_enable_clocks(fe_cl); - if (ret < 0) + if (ret < 0) { + mutex_unlock(&mca->port_mutex); return ret; + } } =20 cl->clocks_in_use[substream->stream] =3D true; @@ -350,6 +360,11 @@ static int mca_be_hw_free(struct snd_pcm_substream *su= bstream, if (cl->port_driver < 0) return -EINVAL; =20 + /* + * We are operating on a foreign cluster here, but since we + * belong to the same PCM, accesses should have been + * synchronized at ASoC level. + */ fe_cl =3D &mca->clusters[cl->port_driver]; if (!mca_fe_clocks_in_use(fe_cl)) return 0; /* Nothing to do */ @@ -722,7 +737,9 @@ static int mca_be_startup(struct snd_pcm_substream *sub= stream, cl->base + REG_PORT_CLOCK_SEL); writel_relaxed(PORT_DATA_SEL_TXA(fe_cl->no), cl->base + REG_PORT_DATA_SEL); + mutex_lock(&mca->port_mutex); cl->port_driver =3D fe_cl->no; + mutex_unlock(&mca->port_mutex); cl->port_started[substream->stream] =3D true; =20 return 0; @@ -732,6 +749,7 @@ static void mca_be_shutdown(struct snd_pcm_substream *s= ubstream, struct snd_soc_dai *dai) { struct mca_cluster *cl =3D mca_dai_to_cluster(dai); + struct mca_data *mca =3D cl->host; =20 cl->port_started[substream->stream] =3D false; =20 @@ -742,7 +760,9 @@ static void mca_be_shutdown(struct snd_pcm_substream *s= ubstream, */ writel_relaxed(0, cl->base + REG_PORT_ENABLES); writel_relaxed(0, cl->base + REG_PORT_DATA_SEL); + mutex_lock(&mca->port_mutex); cl->port_driver =3D -1; + mutex_unlock(&mca->port_mutex); } } =20 @@ -963,6 +983,7 @@ static int apple_mca_probe(struct platform_device *pdev) return -ENOMEM; mca->dev =3D &pdev->dev; mca->nclusters =3D nclusters; + mutex_init(&mca->port_mutex); platform_set_drvdata(pdev, mca); clusters =3D mca->clusters; =20 --=20 2.33.0