Received: by 10.213.65.68 with SMTP id h4csp431797imn; Fri, 6 Apr 2018 02:50:36 -0700 (PDT) X-Google-Smtp-Source: AIpwx4/E0SKL3PnMZjrT4WfP8j1Z91WAf6OvT0UuMKGxetNdspFM+ysdXQADYK6WwVHHJ0NhvcbP X-Received: by 10.99.2.202 with SMTP id 193mr17396540pgc.117.1523008236210; Fri, 06 Apr 2018 02:50:36 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1523008236; cv=none; d=google.com; s=arc-20160816; b=eT8awi1Lf/Ajik3Fv9Uj2aBWQeoqNVW1ze6J+6VpwORUDLz2GtsN46q2nZ6Gof3SW+ +S+bbEapHYPDTtsNEOupH0X9i0KJCno0yFvdfxLz08kRTDQZbRU9piHpu35xrJSg2CkG tjMgsWjiAvijRAQ5ag4VdD7MVZQkhI26XjQ9etfvnfEOdCyaIHIzyNib9dEH/OKnbqNq H+yrPvV4XU9nRc1NBRp2ipwWQZRAAQNkfIpqxTWGQslaMjiJ5fsvujAVSdc3RnhrmK5c bCnrY9zNXKZAM4s3BFMyglgRVNfhB6hSq6LfLUT0iPqE6ufckmRQQUAzO3EnokM2MAhS wvfQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding :content-language:in-reply-to:mime-version:user-agent:date :message-id:from:references:cc:to:subject:dmarc-filter :dkim-signature:dkim-signature:arc-authentication-results; bh=zPcPJZH3pvmJQPEYAUsnWyNFCwr/RtlypI7SPa5U1pE=; b=tYyoCUiTd5ADOmRlZS32Ayl58R2L9JaTIWfmtt2EXV9cRzyD6HmQ3nEoaPMf1/5p51 0SATJc+E8NPa0e5LYs2qQFB7PpkggUiYqwZFd9TbNwN8WobLdjZEkDn4Cryz1kgzN4NY /rS+eyi1xAqQF63CIFKyt4f4BGR9vKfQvVxiYMXqpUdOmJFtQwuqHaFomHlVGRX6teX6 EwCP6QTY2kYDVIZAA8x9NXRf3fmMPOcC192XXYGpyyVXPRf5Lj3jTqaWUJeGFyjhuw1m bTejmnOKGkYdEYnBoHKs75cPsJyX7jcBkrCNt/OFwfuIeZ52+rU+Ax4pnoH934mF6pCg 19Qg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@codeaurora.org header.s=default header.b=olk2IAoI; dkim=pass header.i=@codeaurora.org header.s=default header.b=lnkYH3YY; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id g1-v6si7807780plt.520.2018.04.06.02.50.22; Fri, 06 Apr 2018 02:50:36 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@codeaurora.org header.s=default header.b=olk2IAoI; dkim=pass header.i=@codeaurora.org header.s=default header.b=lnkYH3YY; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1751974AbeDFJs7 (ORCPT + 99 others); Fri, 6 Apr 2018 05:48:59 -0400 Received: from smtp.codeaurora.org ([198.145.29.96]:49584 "EHLO smtp.codeaurora.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1751454AbeDFJs5 (ORCPT ); Fri, 6 Apr 2018 05:48:57 -0400 Received: by smtp.codeaurora.org (Postfix, from userid 1000) id D490860850; Fri, 6 Apr 2018 09:48:56 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=codeaurora.org; s=default; t=1523008136; bh=9EHxx7BkGtvMMvtLdbQVoeiaeQjdhTi0wMMMBBIcZ9k=; h=Subject:To:Cc:References:From:Date:In-Reply-To:From; b=olk2IAoIehs7fJPxv0+OqBTvMVmzKwacXenf/zaxRR4sbos+DP7t+OyojPIRooI/7 Djgpk/6/u18/Z8tmCWLlgqzyBavuSG9+hZBE5FT3Bu5usJZ+o0QFDMRODEGIQ+fHBO HH1YdB6boaIdfKpzaPgKoCgi+vTbKmafurL9asz4= X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on pdx-caf-mail.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-2.8 required=2.0 tests=ALL_TRUSTED,BAYES_00, DKIM_SIGNED,T_DKIM_INVALID autolearn=no autolearn_force=no version=3.4.0 Received: from [10.206.25.30] (blr-c-bdr-fw-01_globalnat_allzones-outside.qualcomm.com [103.229.19.19]) (using TLSv1.2 with cipher ECDHE-RSA-AES128-GCM-SHA256 (128/128 bits)) (No client certificate requested) (Authenticated sender: vviswana@smtp.codeaurora.org) by smtp.codeaurora.org (Postfix) with ESMTPSA id 2FB7F601B4; Fri, 6 Apr 2018 09:48:46 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=codeaurora.org; s=default; t=1523008135; bh=9EHxx7BkGtvMMvtLdbQVoeiaeQjdhTi0wMMMBBIcZ9k=; h=Subject:To:Cc:References:From:Date:In-Reply-To:From; b=lnkYH3YYNe0k2ZfiJqeGxRSUA2H07vKw8OUBRkvL5kFE9noVDpB4s6pV6c1n9jz83 uuug+Rb78I4LdtH/kL6lszK0SuSJDmJrxl8KBgA3vr6ptwLQ3/MZItsQQogQBqMayo NhMgFpdXyXMKqb5ksGMVxWHXo5fMWxmBi6jzEBMM= DMARC-Filter: OpenDMARC Filter v1.3.2 smtp.codeaurora.org 2FB7F601B4 Authentication-Results: pdx-caf-mail.web.codeaurora.org; dmarc=none (p=none dis=none) header.from=codeaurora.org Authentication-Results: pdx-caf-mail.web.codeaurora.org; spf=none smtp.mailfrom=vviswana@codeaurora.org Subject: Re: [PATCH V4 2/2] mmc: sdhci-msm: support voltage pad switching To: Doug Anderson Cc: Adrian Hunter , Ulf Hansson , linux-mmc@vger.kernel.org, LKML , Shawn Lin , linux-arm-msm@vger.kernel.org, georgi.djakov@linaro.org, asutoshd@codeaurora.org, stummala@codeaurora.org, venkatg@codeaurora.org, pramod.gurav@linaro.org, jeremymc@redhat.com, Bjorn Andersson , riteshh@codeaurora.org, Krishna Konda , Asutosh Das References: <1522242500-10556-1-git-send-email-vviswana@codeaurora.org> <1522242500-10556-3-git-send-email-vviswana@codeaurora.org> From: Vijay Viswanath Message-ID: <637147e3-5006-3593-3b01-9516375f8995@codeaurora.org> Date: Fri, 6 Apr 2018 15:18:43 +0530 User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64; rv:52.0) Gecko/20100101 Thunderbird/52.7.0 MIME-Version: 1.0 In-Reply-To: Content-Type: text/plain; charset=utf-8; format=flowed Content-Language: en-US Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 3/29/2018 4:23 AM, Doug Anderson wrote: > Hi, > > On Wed, Mar 28, 2018 at 6:08 AM, Vijay Viswanath > wrote: >> From: Krishna Konda >> >> The PADs for SD card are dual-voltage that support 3v/1.8v. Those PADs >> have a control signal (io_pad_pwr_switch/mode18 ) that indicates >> whether the PAD works in 3v or 1.8v. >> >> SDHC core on msm platforms should have IO_PAD_PWR_SWITCH bit set/unset >> based on actual voltage used for IO lines. So when power irq is >> triggered for io high or io low, the driver should check the voltages >> supported and set the pad accordingly. >> >> Signed-off-by: Krishna Konda >> Signed-off-by: Venkat Gopalakrishnan >> Signed-off-by: Vijay Viswanath >> --- >> drivers/mmc/host/sdhci-msm.c | 64 ++++++++++++++++++++++++++++++++++++++++++-- >> 1 file changed, 62 insertions(+), 2 deletions(-) >> >> diff --git a/drivers/mmc/host/sdhci-msm.c b/drivers/mmc/host/sdhci-msm.c >> index 2fcd9010..bbf9626 100644 >> --- a/drivers/mmc/host/sdhci-msm.c >> +++ b/drivers/mmc/host/sdhci-msm.c >> @@ -78,12 +78,15 @@ >> #define CORE_HC_MCLK_SEL_DFLT (2 << 8) >> #define CORE_HC_MCLK_SEL_HS400 (3 << 8) >> #define CORE_HC_MCLK_SEL_MASK (3 << 8) >> +#define CORE_IO_PAD_PWR_SWITCH_EN (1 << 15) >> +#define CORE_IO_PAD_PWR_SWITCH (1 << 16) >> #define CORE_HC_SELECT_IN_EN BIT(18) >> #define CORE_HC_SELECT_IN_HS400 (6 << 19) >> #define CORE_HC_SELECT_IN_MASK (7 << 19) >> >> #define CORE_3_0V_SUPPORT (1 << 25) >> #define CORE_1_8V_SUPPORT (1 << 26) >> +#define CORE_VOLT_SUPPORT (CORE_3_0V_SUPPORT | CORE_1_8V_SUPPORT) >> >> #define CORE_CSR_CDC_CTLR_CFG0 0x130 >> #define CORE_SW_TRIG_FULL_CALIB BIT(16) >> @@ -1109,7 +1112,7 @@ static void sdhci_msm_handle_pwr_irq(struct sdhci_host *host, int irq) >> u32 irq_status, irq_ack = 0; >> int retry = 10; >> u32 pwr_state = 0, io_level = 0; >> - >> + u32 config; >> >> irq_status = readl_relaxed(msm_host->core_mem + CORE_PWRCTL_STATUS); >> irq_status &= INT_MASK; >> @@ -1166,6 +1169,45 @@ static void sdhci_msm_handle_pwr_irq(struct sdhci_host *host, int irq) >> */ >> writel_relaxed(irq_ack, msm_host->core_mem + CORE_PWRCTL_CTL); >> >> + /* >> + * If we don't have info regarding the voltage levels supported by >> + * regulators, don't change the IO PAD PWR SWITCH. >> + */ >> + if (msm_host->caps_0 & CORE_VOLT_SUPPORT) { >> + /* Ensure order between core_mem and hc_mem */ >> + mb(); > > Like in v2, I don't understand why you need a mb() before the read > from CORE_VENDOR_SPEC. No reads or writes to the core_mem will affect > the value you're reading here, so you need no barrier. > > If you need a barrier before the _write_ to CORE_VENDOR_SPEC then add > it below. Then in the case where the config doesn't change you have > no barriers. > > >> + /* >> + * We should unset IO PAD PWR switch only if the register write >> + * can set IO lines high and the regulator also switches to 3 V. >> + * Else, we should keep the IO PAD PWR switch set. >> + * This is applicable to certain targets where eMMC vccq supply >> + * is only 1.8V. In such targets, even during REQ_IO_HIGH, the >> + * IO PAD PWR switch must be kept set to reflect actual >> + * regulator voltage. This way, during initialization of >> + * controllers with only 1.8V, we will set the IO PAD bit >> + * without waiting for a REQ_IO_LOW. >> + */ > > For the above comment, what about just: > > new_config = config > if (msm_host->caps_0 == CORE_1_8V_SUPPORT) { > new_config |= CORE_IO_PAD_PWR_SWITCH; > } else if (msm_host->caps_0 == CORE_3_3V_SUPPORT) { > new_config &= ~CORE_IO_PAD_PWR_SWITCH; > } else if (msm_host->caps_0 & CORE_VOLT_SUPPORT) { > if (io_level & REQ_IO_HIGH) > new_config &= ~CORE_IO_PAD_PWR_SWITCH; > else if (io_level & REQ_IO_LOW) > new_config |= CORE_IO_PAD_PWR_SWITCH; > } This looks a big mess of if/else. Does the above implementation have better performance compared to having two if/else with bit operations inside ? The latter looks much cleaner and faster. If regulator only supports 3V and we get a io_low from BUS_OFF ( REQ_IO_LOW should never come if we don't support 1.8V), it is ok to set io pad. > if (config != new_config) { > ... > } > > AKA: first check if it only supports one voltage and pick that one. > Else if it supports both you can use the request. This might be more > important if you get rid of the initial setting in > sdhci_msm_set_regulator_caps() as I'm suggesting. > > >> + config = readl_relaxed(host->ioaddr + CORE_VENDOR_SPEC); >> + >> + if (((io_level & REQ_IO_HIGH) && (msm_host->caps_0 & >> + CORE_3_0V_SUPPORT)) && >> + (config & CORE_IO_PAD_PWR_SWITCH)) { >> + config &= ~CORE_IO_PAD_PWR_SWITCH; >> + writel_relaxed(config, >> + host->ioaddr + CORE_VENDOR_SPEC); >> + /* IO PAD register is in different memory space */ >> + mb(); > > Wow, for a driver that tries so hard to use "relaxed" versions of > writes to avoid barriers you sure end up needing to sprinkle a lot of > these around "just in case". :( ...this one seems extra fishy > because: > > * There are no more accesses after this one in this function. > > * If you're worried about something that happens outside of the > context of the IRQ needing this wb() then that's a silly concern. > Presumably if they were doing anything that could race with you they'd > have a lock and locking routines are implicit barriers. > > * In the context of the IRQ itself the next call is > sdhci_msm_complete_pwr_irq_wait(), which eventually calls wake_up. > This has a locking primitive and thus an implicit barrier. > Sorry, I didn't get implicit barrier in locking primitive. In mmc_set_signal_voltage switch: 1. Send cmd 11. 2. Switch 1.8V in SDHCI_HOST_CONTROL2 3. Wait for pwr_irq wake_up(). 4. pwr_irq context comes up & does register read/writes in core mem. Updates IO PAD in HC mem. 5. pwr_irq calls wake_up. 6. mmc_set_signal_voltage_switch context does further register read/writes which expects IO_PAD change within pwr_irq context is complete before step 6. Can wake_up() ensure that any update to CORE_VENDOR_SPEC happens before any register writes in HC after the wake_up() ? > * There's a direct call of sdhci_msm_handle_pwr_irq() from probe, and > it has a big fat mb(). I have a hard time believing that matters too > because I'd bet "platform_get_irq_byname" has at least one lock in it. > > > IMHO these "_relaxed" calls are just not worth it except in _very_ > targeted usage. > > >> + } else if (((io_level & REQ_IO_LOW) || >> + (msm_host->caps_0 & CORE_1_8V_SUPPORT)) && >> + !(config & CORE_IO_PAD_PWR_SWITCH)) { >> + config |= CORE_IO_PAD_PWR_SWITCH; >> + writel_relaxed(config, >> + host->ioaddr + CORE_VENDOR_SPEC); >> + /* IO PAD bit is in different memory space */ >> + mb(); >> + } >> + } >> + >> if (pwr_state) >> msm_host->curr_pwr_state = pwr_state; >> if (io_level) >> @@ -1322,7 +1364,8 @@ static int sdhci_msm_set_regulator_caps(struct sdhci_msm_host *msm_host) >> { >> struct mmc_host *mmc = msm_host->mmc; >> struct regulator *supply = mmc->supply.vqmmc; >> - u32 caps = 0; >> + u32 caps = 0, config; >> + struct sdhci_host *host = mmc_priv(mmc); >> >> if (!IS_ERR(mmc->supply.vqmmc)) { >> if (regulator_is_supported_voltage(supply, 1700000, 1950000)) >> @@ -1335,6 +1378,23 @@ static int sdhci_msm_set_regulator_caps(struct sdhci_msm_host *msm_host) >> mmc_hostname(mmc), __func__); >> } >> >> + if (caps) { >> + /* >> + * Set the PAD_PWR_SWITCH_EN bit so that the PAD_PWR_SWITCH >> + * bit can be used as required later on. >> + */ >> + u32 io_level = msm_host->curr_io_level; >> + >> + config = readl_relaxed(host->ioaddr + CORE_VENDOR_SPEC); >> + config |= CORE_IO_PAD_PWR_SWITCH_EN; >> + >> + if ((io_level & REQ_IO_HIGH) && (caps & CORE_3_0V_SUPPORT)) > > Slight nit that there's a tab character after "caps &". Please > replace it with a space. > > Will do >> + config &= ~CORE_IO_PAD_PWR_SWITCH; >> + else if ((io_level & REQ_IO_LOW) || (caps & CORE_1_8V_SUPPORT)) >> + config |= CORE_IO_PAD_PWR_SWITCH; > > Are you sure that's right? In English: > > * If we requested high and we support high then set to high. > * else if we requested low __or__ we support low then set low. > > Things that are weird above that: > > * If we request low but don't support low, switch to low anyway. > * If we request high but only support low, switch to low anyway. > > If nothing else seems like this would deserve a comment, but I'd be > curious of the justification for that logic. > > > Also: seems like this is duplicated code between here and > sdhci_msm_handle_pwr_irq(). Does it even need to be here? Can't you > just move the call to sdhci_msm_set_regulator_caps() before the call > to sdhci_msm_handle_pwr_irq() in probe? Then just let that first call > to to sdhci_msm_handle_pwr_irq() do this work? In > sdhci_msm_handle_pwr_irq() you can always just "OR" in > CORE_IO_PAD_PWR_SWITCH_EN > > > -Doug > -- If we don't support 1.8V, then the only time io_low will happen is during BUS_OFF. For BUS_OFF, enabling IO_PAD_PWR_SWITCH is ok. This logic is same as what is there in pwr_irq. Added the same stuff here because by the time we get regulator info from mmc layer, some power irqs would have already come and gone. > To unsubscribe from this list: send the line "unsubscribe linux-mmc" in > the body of a message to majordomo@vger.kernel.org > More majordomo info at http://vger.kernel.org/majordomo-info.html >