Received: by 2002:a05:6a10:6d10:0:0:0:0 with SMTP id gq16csp776431pxb; Fri, 22 Apr 2022 10:55:29 -0700 (PDT) X-Google-Smtp-Source: ABdhPJwMdQosWyvetg4pswXzMYxkoKNpRFuSI8akp4SQTOb1/Mr0i5LTLusc5JvA6uXoki/57lks X-Received: by 2002:a17:902:b941:b0:14d:af72:3f23 with SMTP id h1-20020a170902b94100b0014daf723f23mr5898316pls.6.1650650128944; Fri, 22 Apr 2022 10:55:28 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1650650128; cv=none; d=google.com; s=arc-20160816; b=jpZUiyjW9JNWiGHeK7MYj3D/2H0yvIfQvVge+IIOyODnYINFQQ/B+04o5QNaKS6nvJ nKO/IVE9jtZh/QVMlx4rumJyI34dNf1X95IIu2JnSk3AD3diKloRyJQmCtuCxLtwymIO rTXSLdGDNmPJfQy4wk9DHh6UbXb+eGikbf42u+wRXs4H9XYWwBZYvf1fd7aZzaqb8RzH VIzMgC46NAYp9xx6tsjUXkS7bNOjFWKYqxE3uhzTH7MN7vqYhErkk1PSyF7bYwSSctN6 VyS/CJpSIxZu13FuQwSi9Esx/VPsSdUtu27YKO1dSvrGpbIEalbC2a4Ks0UYQaEW3SpX AeIA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:message-id:date:subject:cc:to:from :dkim-signature; bh=pR5DqmLIfQR3RKTh/d51Lo5eOrMeqk/ZSkl9py1PXdA=; b=v81GuqQbCx0qd449vY1ZTg0JvFBj+6eRB8PyqIUxFp03t/DptGI+1OPh/9NagPZv+7 UJDay88XaZKsup8CauvzMsD59mWnRVoNf2Y5LzAH/iVjsIiMdZq7mD1R8cE+dvCWX9D9 K5TjgG7dkL5NRQw03xa6/nVTtKwkEK7ebeUhECIevnnZsuPT4a85+6m1oaT4SHG+r3a3 A91omjnP/LsWEREy7B5MkOoXj9DGjuLKOTf14UYwAjrC+zBKtLjZGyL82W5nKfko1Na8 iKqQmcuCtdiMKDNb2eEFicJsK374ghywjuEbmWbyRnoOz42E0ZLqYNo1SfXHuByX+OJB NQ6Q== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@intel.com header.s=Intel header.b=Ceb8FsFw; spf=softfail (google.com: domain of transitioning linux-kernel-owner@vger.kernel.org does not designate 23.128.96.19 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=intel.com Return-Path: Received: from lindbergh.monkeyblade.net (lindbergh.monkeyblade.net. [23.128.96.19]) by mx.google.com with ESMTPS id m4-20020a632604000000b0039cdd7341b2si8947413pgm.601.2022.04.22.10.55.28 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 22 Apr 2022 10:55:28 -0700 (PDT) Received-SPF: softfail (google.com: domain of transitioning linux-kernel-owner@vger.kernel.org does not designate 23.128.96.19 as permitted sender) client-ip=23.128.96.19; Authentication-Results: mx.google.com; dkim=pass header.i=@intel.com header.s=Intel header.b=Ceb8FsFw; spf=softfail (google.com: domain of transitioning linux-kernel-owner@vger.kernel.org does not designate 23.128.96.19 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=intel.com Received: from out1.vger.email (out1.vger.email [IPv6:2620:137:e000::1:20]) by lindbergh.monkeyblade.net (Postfix) with ESMTP id 62DB110782D; Fri, 22 Apr 2022 10:38:43 -0700 (PDT) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1348628AbiDTCdi (ORCPT + 99 others); Tue, 19 Apr 2022 22:33:38 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:57696 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1348564AbiDTCdg (ORCPT ); Tue, 19 Apr 2022 22:33:36 -0400 Received: from mga12.intel.com (mga12.intel.com [192.55.52.136]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id D790811161 for ; Tue, 19 Apr 2022 19:30:49 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=intel.com; i=@intel.com; q=dns/txt; s=Intel; t=1650421849; x=1681957849; h=from:to:cc:subject:date:message-id; bh=2x5p0qshyNJJblwHIKzMWnvRCa7FvV22Ri0Wm88vg/s=; b=Ceb8FsFw+OB0nGb7cFtOAqLPqVbfUB5Lng0+WpWJ+7Wpl+b+eNnVDYAR 9R36Yf+/bFOsJTBF/H2I/dOAXTywht1jQn8yYrabtKTbq0Eo3xIyy3scv tVx4RBxoYgnwG2NePbIrTQ2Q0ntVuaoy0middT1vOxs+kJGm3Ryi2pjQP HwwzUELly8SWh2qjVN0fCu6QIXwfC81fbfG2EDJDH01HJOEkrvliduCtB k33muiVm1VUIg6QwDRxtE+S/LoV5BbO7D3OPNSozTWCmfcBvxxiN9SatA 0+rKRnKEeuzS42tbzhRUZqPOTcE9ifI+hdHqbmZhZq8cjZnD+iSOvwW87 w==; X-IronPort-AV: E=McAfee;i="6400,9594,10322"; a="243852890" X-IronPort-AV: E=Sophos;i="5.90,274,1643702400"; d="scan'208";a="243852890" Received: from orsmga006.jf.intel.com ([10.7.209.51]) by fmsmga106.fm.intel.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 19 Apr 2022 19:30:49 -0700 X-IronPort-AV: E=Sophos;i="5.90,274,1643702400"; d="scan'208";a="529554083" Received: from bard-ubuntu.sh.intel.com ([10.239.185.57]) by orsmga006-auth.jf.intel.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 19 Apr 2022 19:30:46 -0700 From: Bard Liao To: alsa-devel@alsa-project.org, vkoul@kernel.org Cc: vinod.koul@linaro.org, linux-kernel@vger.kernel.org, gregkh@linuxfoundation.org, srinivas.kandagatla@linaro.org, pierre-louis.bossart@linux.intel.com, sanyog.r.kale@intel.com, bard.liao@intel.com Subject: [PATCH] soundwire: cadence: recheck device0 attachment after status change Date: Wed, 20 Apr 2022 10:30:39 +0800 Message-Id: <20220420023039.14144-1-yung-chuan.liao@linux.intel.com> X-Mailer: git-send-email 2.17.1 X-Spam-Status: No, score=-2.4 required=5.0 tests=BAYES_00,DKIMWL_WL_HIGH, DKIM_SIGNED,DKIM_VALID,HEADER_FROM_DIFFERENT_DOMAINS, MAILING_LIST_MULTI,RDNS_NONE,SPF_HELO_NONE autolearn=unavailable autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Pierre-Louis Bossart This patch adds a status check after device0 attachment to solve race conditions observed during attachment with multiple devices per link The sequence is the following 1) deviceA attaches as device0 2) the hardware detects a device0 status change and throws an interrupt. 3) the interrupt handler schedules the work function 4) the workqueue starts, we read the status slave0 = cdns_readl(cdns, CDNS_MCP_SLAVE_INTSTAT0); slave1 = cdns_readl(cdns, CDNS_MCP_SLAVE_INTSTAT1); we deal with the status change and program deviceA device number to a non-zero value. 5) deviceB attaches as device0, the device0 status seen by the hardware does not change. 6) we clear the CDNS_MCP_SLAVE_INTSTAT0/1 registers -> we will never detect deviceB! This patch suggest re-checking in a loop the device0 status with a PING frame, i.e. using the real device0 status instead of information on status changes. Signed-off-by: Pierre-Louis Bossart Reviewed-by: Rander Wang Signed-off-by: Bard Liao --- drivers/soundwire/cadence_master.c | 37 ++++++++++++++++++++++++++++++ 1 file changed, 37 insertions(+) diff --git a/drivers/soundwire/cadence_master.c b/drivers/soundwire/cadence_master.c index 558390af44b6..47d59190a96e 100644 --- a/drivers/soundwire/cadence_master.c +++ b/drivers/soundwire/cadence_master.c @@ -959,6 +959,8 @@ static void cdns_update_slave_status_work(struct work_struct *work) container_of(work, struct sdw_cdns, work); u32 slave0, slave1; u64 slave_intstat; + u32 device0_status; + int retry_count = 0; slave0 = cdns_readl(cdns, CDNS_MCP_SLAVE_INTSTAT0); slave1 = cdns_readl(cdns, CDNS_MCP_SLAVE_INTSTAT1); @@ -968,10 +970,45 @@ static void cdns_update_slave_status_work(struct work_struct *work) dev_dbg_ratelimited(cdns->dev, "Slave status change: 0x%llx\n", slave_intstat); +update_status: cdns_update_slave_status(cdns, slave_intstat); cdns_writel(cdns, CDNS_MCP_SLAVE_INTSTAT0, slave0); cdns_writel(cdns, CDNS_MCP_SLAVE_INTSTAT1, slave1); + /* + * When there is more than one peripheral per link, it's + * possible that a deviceB becomes attached after we deal with + * the attachment of deviceA. Since the hardware does a + * logical AND, the attachment of the second device does not + * change the status seen by the driver. + * + * In that case, clearing the registers above would result in + * the deviceB never being detected - until a change of status + * is observed on the bus. + * + * To avoid this race condition, re-check if any device0 needs + * attention with PING commands. There is no need to check for + * ALERTS since they are not allowed until a non-zero + * device_number is assigned. + */ + + device0_status = cdns_readl(cdns, CDNS_MCP_SLAVE_STAT); + device0_status &= 3; + + if (device0_status == SDW_SLAVE_ATTACHED) { + if (retry_count++ < SDW_MAX_DEVICES) { + dev_dbg_ratelimited(cdns->dev, + "Device0 detected after clearing status, iteration %d\n", + retry_count); + slave_intstat = CDNS_MCP_SLAVE_INTSTAT_ATTACHED; + goto update_status; + } else { + dev_err_ratelimited(cdns->dev, + "Device0 detected after %d iterations\n", + retry_count); + } + } + /* clear and unmask Slave interrupt now */ cdns_writel(cdns, CDNS_MCP_INTSTAT, CDNS_MCP_INT_SLAVE_MASK); cdns_updatel(cdns, CDNS_MCP_INTMASK, -- 2.17.1