Message ID | 20221213090047.3805-15-victor.shih@genesyslogic.com.tw |
---|---|
State | New |
Headers |
Return-Path: <linux-kernel-owner@vger.kernel.org> Delivered-To: ouuuleilei@gmail.com Received: by 2002:a05:7300:8188:b0:89:790f:f786 with SMTP id m8csp3858038dye; Tue, 13 Dec 2022 01:04:35 -0800 (PST) X-Google-Smtp-Source: AA0mqf6EacH5n6K1gOQqOjAnK+wE4HDg/kFu+WOQt3EZIdPm6pRAoZ7aIBJE8eftLGYf5bK+o73F X-Received: by 2002:a05:6a21:9996:b0:a2:491c:15cc with SMTP id ve22-20020a056a21999600b000a2491c15ccmr29146545pzb.40.1670922274963; Tue, 13 Dec 2022 01:04:34 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1670922274; cv=none; d=google.com; s=arc-20160816; b=hsxUxTGukYn1DeF04TpdP9hljF0V39mnHdfk7EobiptjenHXPkvo0HOcO/qlrO5YEQ WhOD3ptKpmG673NkhI4UmqLp1VkY3tnpjt2gWMLUi5yja27SBu2leMaLHP7cia40KQxt jg6plPFdCzQgRZju4JVlnZYNxhnG0ee+hPapiTAtmEUjHiqnmcDYTmmw1X/p6LlQOh17 L8egeKc1VEPrJGeo/QRS5CxAMx3JVQbzjlstDzEtpjyIShPz4FqxC8iQUBR9tLTRsZZJ l7iCBy9mPel5u2Rd4Xw6iS3sRvU51fDQz4BiJSi87+GKTO0nBElqC5Y2xtFwt5qDiUFd C0ew== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from :dkim-signature; bh=pDw/SHDyLtu1AoZL+hYcNtOe2c/7a7zkWUntPhiSPcU=; b=Bk1VAi00+lMTSJwCAAWCmNhLeM2E84Jvvsy5pjlxxskrxdHWIxmcUsmECvig2q9ocq NFfA5mAs8YPTNDu1TH6+Rw36I2YeGkZheWGe+GuUwTbzXVoF6nPqaxldxo2hhjuKEdr0 Tho6CA3EocmmTxj3OI/keuim+aZZeKLAsOG4Q2YsNN+0wf6FUou8uWfK6aD1y+E5MvhU oB2k7dpGp8Oz0kYB/w/EhSnL28Yi71yvsRJX5IDaL4tAbE7AB4eCRofdOAbg1ozp+vTw vt3PHfEuPLBVX3k19PUXHMGn3GpEVlNj0/b25thLjylr8ISwobZCr7sdDBti/577alQY Y9CQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@gmail.com header.s=20210112 header.b=glOVOtQi; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id m187-20020a6326c4000000b00476e9888164si12156562pgm.88.2022.12.13.01.04.21; Tue, 13 Dec 2022 01:04:34 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; dkim=pass header.i=@gmail.com header.s=20210112 header.b=glOVOtQi; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S235054AbiLMJDf (ORCPT <rfc822;jeantsuru.cumc.mandola@gmail.com> + 99 others); Tue, 13 Dec 2022 04:03:35 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:36628 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S235002AbiLMJCG (ORCPT <rfc822;linux-kernel@vger.kernel.org>); Tue, 13 Dec 2022 04:02:06 -0500 Received: from mail-pg1-x535.google.com (mail-pg1-x535.google.com [IPv6:2607:f8b0:4864:20::535]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 693B4E0B9; Tue, 13 Dec 2022 01:01:54 -0800 (PST) Received: by mail-pg1-x535.google.com with SMTP id r18so9967256pgr.12; Tue, 13 Dec 2022 01:01:54 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=content-transfer-encoding:mime-version:references:in-reply-to :message-id:date:subject:cc:to:from:from:to:cc:subject:date :message-id:reply-to; bh=pDw/SHDyLtu1AoZL+hYcNtOe2c/7a7zkWUntPhiSPcU=; b=glOVOtQidwJKZVZBqcD+XBJEfihYgs1NMCKfbLh26RUGAelFNYgndurIJZHAHVuuHU HbFH0sLsDFR35Qf9M0939ijWNRArbOcj+6oqcQ6Rf/lyCpxb+keoAI8C6hpS9t+gZVZA UQmh6yeT5o2kfbzoK/eP3HhipJbnF67kk7NFKL2mN3GqaqCru050nrfMEc5w3Gx+0/jX mG/1IuaMDNVyhKkkN9a+qaTL9c6C8yn5OSkcoPAhXqtiMD1w4wF8JNktHTtNrsyom1KX XKXbRrXN60PGpEBpbo7lgF6AeWxX0zb3GnvRQlgpOIuZwUVoQRpSE4anYlSU1aPO7lHs yzPQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=content-transfer-encoding:mime-version:references:in-reply-to :message-id:date:subject:cc:to:from:x-gm-message-state:from:to:cc :subject:date:message-id:reply-to; bh=pDw/SHDyLtu1AoZL+hYcNtOe2c/7a7zkWUntPhiSPcU=; b=xblr2iQIc3skqU+isvYfFsy1C8sQwZ4QF7XpBf7pf7jHOjXVHGZqbJp61U/US4WrTk vCESzp5QMEhLaMoabjFcIk+nmxfLuaz6U44x5+6wUPXaJB+WHFksmv0P4DfZpL/uZRes Ox3G+NuwD8vRJeHpdpyAuKPUpmwbn6SygJUZ3mVaMijTJnLJBOkHj0z/eZ1W8ME3WI2r KsWS/1By7koUNq8JLd+2ujmQ3iANqUxdRd8KHXKCRQtC6zTmI2Esbv4GWTR8GQvuk58f P/mSZITPv0+0eJFPG87yM8bV+6xdZQkUZPA2+LVcMjchbIW0eN9zhDPsM4lCnpdimM01 Qcpg== X-Gm-Message-State: ANoB5pmON9yzSFRvqA3GZ0FAx3dRdVywddOuLEp21x9xrEr7+kBOZ966 omreyCcBZPgAsmj1Y8SK0/4= X-Received: by 2002:a62:e519:0:b0:575:bf29:d1e0 with SMTP id n25-20020a62e519000000b00575bf29d1e0mr21516224pff.26.1670922113880; Tue, 13 Dec 2022 01:01:53 -0800 (PST) Received: from localhost.localdomain (2001-b400-e2dc-5c03-264b-f684-a8c9-21ae.emome-ip6.hinet.net. [2001:b400:e2dc:5c03:264b:f684:a8c9:21ae]) by smtp.gmail.com with ESMTPSA id 81-20020a621954000000b0056b9ec7e2desm7141682pfz.125.2022.12.13.01.01.51 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Tue, 13 Dec 2022 01:01:53 -0800 (PST) From: Victor Shih <victorshihgli@gmail.com> X-Google-Original-From: Victor Shih <victor.shih@genesyslogic.com.tw> To: ulf.hansson@linaro.org, adrian.hunter@intel.com Cc: linux-mmc@vger.kernel.org, linux-kernel@vger.kernel.org, benchuanggli@gmail.com, HL.Liu@genesyslogic.com.tw, Greg.tu@genesyslogic.com.tw, takahiro.akashi@linaro.org, dlunev@chromium.org, Victor Shih <victor.shih@genesyslogic.com.tw>, Ben Chuang <ben.chuang@genesyslogic.com.tw> Subject: [PATCH V6 14/24] mmc: sdhci-uhs2: add set_ios() Date: Tue, 13 Dec 2022 17:00:37 +0800 Message-Id: <20221213090047.3805-15-victor.shih@genesyslogic.com.tw> X-Mailer: git-send-email 2.25.1 In-Reply-To: <20221213090047.3805-1-victor.shih@genesyslogic.com.tw> References: <20221213090047.3805-1-victor.shih@genesyslogic.com.tw> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-Spam-Status: No, score=-2.1 required=5.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,FREEMAIL_FROM, RCVD_IN_DNSWL_NONE,SPF_HELO_NONE,SPF_PASS autolearn=ham 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: <linux-kernel.vger.kernel.org> X-Mailing-List: linux-kernel@vger.kernel.org X-getmail-retrieved-from-mailbox: =?utf-8?q?INBOX?= X-GMAIL-THRID: =?utf-8?q?1752088995672266953?= X-GMAIL-MSGID: =?utf-8?q?1752088995672266953?= |
Series |
Add support UHS-II for GL9755
|
|
Commit Message
Victor Shih
Dec. 13, 2022, 9 a.m. UTC
This is a sdhci version of mmc's set_ios operation. It covers both UHS-I and UHS-II. Signed-off-by: Ben Chuang <ben.chuang@genesyslogic.com.tw> Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org> Signed-off-by: Victor Shih <victor.shih@genesyslogic.com.tw> --- drivers/mmc/host/sdhci-uhs2.c | 96 +++++++++++++++++++++++++++++++++++ drivers/mmc/host/sdhci-uhs2.h | 1 + drivers/mmc/host/sdhci.c | 53 +++++++++++-------- drivers/mmc/host/sdhci.h | 2 + 4 files changed, 131 insertions(+), 21 deletions(-)
Comments
On 13/12/22 11:00, Victor Shih wrote: > This is a sdhci version of mmc's set_ios operation. > It covers both UHS-I and UHS-II. > > Signed-off-by: Ben Chuang <ben.chuang@genesyslogic.com.tw> > Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org> > Signed-off-by: Victor Shih <victor.shih@genesyslogic.com.tw> > --- > drivers/mmc/host/sdhci-uhs2.c | 96 +++++++++++++++++++++++++++++++++++ > drivers/mmc/host/sdhci-uhs2.h | 1 + > drivers/mmc/host/sdhci.c | 53 +++++++++++-------- > drivers/mmc/host/sdhci.h | 2 + > 4 files changed, 131 insertions(+), 21 deletions(-) > > diff --git a/drivers/mmc/host/sdhci-uhs2.c b/drivers/mmc/host/sdhci-uhs2.c > index 00b1b69b49ea..3d52d35a91a5 100644 > --- a/drivers/mmc/host/sdhci-uhs2.c > +++ b/drivers/mmc/host/sdhci-uhs2.c > @@ -213,6 +213,68 @@ void sdhci_uhs2_set_timeout(struct sdhci_host *host, struct mmc_command *cmd) > } > EXPORT_SYMBOL_GPL(sdhci_uhs2_set_timeout); > > +/** > + * sdhci_uhs2_clear_set_irqs - set Error Interrupt Status Enable register > + * @host: SDHCI host > + * @clear: bit-wise clear mask > + * @set: bit-wise set mask > + * > + * Set/unset bits in UHS-II Error Interrupt Status Enable register > + */ > +void sdhci_uhs2_clear_set_irqs(struct sdhci_host *host, u32 clear, u32 set) > +{ > + u32 ier; > + > + ier = sdhci_readl(host, SDHCI_UHS2_INT_STATUS_ENABLE); > + ier &= ~clear; > + ier |= set; > + sdhci_writel(host, ier, SDHCI_UHS2_INT_STATUS_ENABLE); > + sdhci_writel(host, ier, SDHCI_UHS2_INT_SIGNAL_ENABLE); > +} > +EXPORT_SYMBOL_GPL(sdhci_uhs2_clear_set_irqs); > + > +static void __sdhci_uhs2_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > +{ > + struct sdhci_host *host = mmc_priv(mmc); > + u8 cmd_res, dead_lock; > + u16 ctrl_2; > + > + /* UHS2 Timeout Control */ > + sdhci_calc_timeout_uhs2(host, &cmd_res, &dead_lock); > + > + /* change to use calculate value */ > + cmd_res |= FIELD_PREP(SDHCI_UHS2_TIMER_CTRL_DEADLOCK_MASK, dead_lock); > + > + sdhci_uhs2_clear_set_irqs(host, > + SDHCI_UHS2_INT_CMD_TIMEOUT | > + SDHCI_UHS2_INT_DEADLOCK_TIMEOUT, > + 0); > + sdhci_writeb(host, cmd_res, SDHCI_UHS2_TIMER_CTRL); > + sdhci_uhs2_clear_set_irqs(host, 0, > + SDHCI_UHS2_INT_CMD_TIMEOUT | > + SDHCI_UHS2_INT_DEADLOCK_TIMEOUT); > + > + /* UHS2 timing */ Please extend comment to include: Note, UHS2 timing is disabled when powering off > + ctrl_2 = sdhci_readw(host, SDHCI_HOST_CONTROL2); > + if (ios->timing == MMC_TIMING_SD_UHS2) > + ctrl_2 |= SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE; > + else > + ctrl_2 &= ~(SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE); > + sdhci_writew(host, ctrl_2, SDHCI_HOST_CONTROL2); > + > + if (!(host->quirks2 & SDHCI_QUIRK2_PRESET_VALUE_BROKEN)) > + sdhci_enable_preset_value(host, true); > + > + if (host->ops->set_power) > + host->ops->set_power(host, ios->power_mode, ios->vdd); > + else > + sdhci_uhs2_set_power(host, ios->power_mode, ios->vdd); sdhci_set_ios_common() already set the power. Are both needed? > + udelay(100); Please add a comment for why this delay is here. > + > + host->timing = ios->timing; Please move this up to where the timing change is. > + sdhci_set_clock(host, host->clock); > +} > + > /*****************************************************************************\ > * * > * MMC callbacks * > @@ -234,6 +296,39 @@ static int sdhci_uhs2_start_signal_voltage_switch(struct mmc_host *mmc, > return sdhci_start_signal_voltage_switch(mmc, ios); > } > > +int sdhci_uhs2_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) Should be static > +{ > + struct sdhci_host *host = mmc_priv(mmc); > + > + if (!(host->version >= SDHCI_SPEC_400) || > + !(host->mmc->flags & MMC_UHS2_SUPPORT && > + host->mmc->caps2 & MMC_CAP2_SD_UHS2)) { This can be just: if (!sdhci_uhs2_mode(host)) { Not sure if this is actually possible? > + sdhci_set_ios(mmc, ios); > + return 0; > + } > + > + if (ios->power_mode == MMC_POWER_UNDEFINED) > + return 0; > + > + if (host->flags & SDHCI_DEVICE_DEAD) { > + if (!IS_ERR(mmc->supply.vmmc) && > + ios->power_mode == MMC_POWER_OFF) > + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); > + if (!IS_ERR_OR_NULL(mmc->supply.vmmc2) && > + ios->power_mode == MMC_POWER_OFF) > + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc2, 0); This can be just: if (ios->power_mode == MMC_POWER_OFF) { mmc_opt_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); mmc_opt_regulator_set_ocr(mmc, mmc->supply.vmmc2, 0); } > + return -1; > + } > + > + host->timing = ios->timing; __sdhci_uhs2_set_ios() does this so it is not needed here. > + > + sdhci_set_ios_common(mmc, ios); > + > + __sdhci_uhs2_set_ios(mmc, ios); > + > + return 0; > +} > + > /*****************************************************************************\ > * * > * Driver init/exit * > @@ -244,6 +339,7 @@ static int sdhci_uhs2_host_ops_init(struct sdhci_host *host) > { > host->mmc_host_ops.start_signal_voltage_switch = > sdhci_uhs2_start_signal_voltage_switch; > + host->mmc_host_ops.uhs2_set_ios = sdhci_uhs2_set_ios; > > return 0; > } > diff --git a/drivers/mmc/host/sdhci-uhs2.h b/drivers/mmc/host/sdhci-uhs2.h > index a58ef19c08aa..184fee80253c 100644 > --- a/drivers/mmc/host/sdhci-uhs2.h > +++ b/drivers/mmc/host/sdhci-uhs2.h > @@ -178,5 +178,6 @@ void sdhci_uhs2_dump_regs(struct sdhci_host *host); > bool sdhci_uhs2_mode(struct sdhci_host *host); > void sdhci_uhs2_reset(struct sdhci_host *host, u16 mask); > void sdhci_uhs2_set_timeout(struct sdhci_host *host, struct mmc_command *cmd); > +void sdhci_uhs2_clear_set_irqs(struct sdhci_host *host, u32 clear, u32 set); > > #endif /* __SDHCI_UHS2_H */ > diff --git a/drivers/mmc/host/sdhci.c b/drivers/mmc/host/sdhci.c > index 99633a3ef549..49bbdc155b2b 100644 > --- a/drivers/mmc/host/sdhci.c > +++ b/drivers/mmc/host/sdhci.c > @@ -47,8 +47,6 @@ > static unsigned int debug_quirks = 0; > static unsigned int debug_quirks2; > > -static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable); > - > static bool sdhci_send_command(struct sdhci_host *host, struct mmc_command *cmd); > > void sdhci_dumpregs(struct sdhci_host *host) > @@ -1877,6 +1875,9 @@ static u16 sdhci_get_preset_value(struct sdhci_host *host) > case MMC_TIMING_MMC_HS400: > preset = sdhci_readw(host, SDHCI_PRESET_FOR_HS400); > break; > + case MMC_TIMING_SD_UHS2: > + preset = sdhci_readw(host, SDHCI_PRESET_FOR_UHS2); > + break; > default: > pr_warn("%s: Invalid UHS-I mode selected\n", > mmc_hostname(host->mmc)); > @@ -2325,24 +2326,9 @@ static bool sdhci_presetable_values_change(struct sdhci_host *host, struct mmc_i > (sdhci_preset_needed(host, ios->timing) || host->drv_type != ios->drv_type); > } > > -void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > +void sdhci_set_ios_common(struct mmc_host *mmc, struct mmc_ios *ios) > { > struct sdhci_host *host = mmc_priv(mmc); > - bool reinit_uhs = host->reinit_uhs; > - bool turning_on_clk = false; > - u8 ctrl; > - > - host->reinit_uhs = false; > - > - if (ios->power_mode == MMC_POWER_UNDEFINED) > - return; > - > - if (host->flags & SDHCI_DEVICE_DEAD) { > - if (!IS_ERR(mmc->supply.vmmc) && > - ios->power_mode == MMC_POWER_OFF) > - mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); > - return; > - } > > /* > * Reset the chip on each power off. > @@ -2359,8 +2345,6 @@ void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > sdhci_enable_preset_value(host, false); > > if (!ios->clock || ios->clock != host->clock) { > - turning_on_clk = ios->clock && !host->clock; > - > host->ops->set_clock(host, ios->clock); > host->clock = ios->clock; > > @@ -2381,6 +2365,32 @@ void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > host->ops->set_power(host, ios->power_mode, ios->vdd); > else > sdhci_set_power(host, ios->power_mode, ios->vdd); > +} > +EXPORT_SYMBOL_GPL(sdhci_set_ios_common); > + > +void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > +{ > + struct sdhci_host *host = mmc_priv(mmc); > + bool reinit_uhs = host->reinit_uhs; > + bool turning_on_clk = false; > + u8 ctrl; > + > + host->reinit_uhs = false; > + > + if (ios->power_mode == MMC_POWER_UNDEFINED) > + return; > + > + if (host->flags & SDHCI_DEVICE_DEAD) { > + if (!IS_ERR(mmc->supply.vmmc) && > + ios->power_mode == MMC_POWER_OFF) > + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); > + return; > + } > + > + sdhci_set_ios_common(mmc, ios); > + > + if (!ios->clock || ios->clock != host->clock) > + turning_on_clk = ios->clock && !host->clock; > > if (host->ops->platform_send_init_74_clocks) > host->ops->platform_send_init_74_clocks(host, ios->power_mode); > @@ -2959,7 +2969,7 @@ int sdhci_execute_tuning(struct mmc_host *mmc, u32 opcode) > } > EXPORT_SYMBOL_GPL(sdhci_execute_tuning); > > -static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) > +void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) > { > /* Host Controller v3.00 defines preset value registers */ > if (host->version < SDHCI_SPEC_300) > @@ -2987,6 +2997,7 @@ static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) > host->preset_enabled = enable; > } > } > +EXPORT_SYMBOL_GPL(sdhci_enable_preset_value); > > static void sdhci_post_req(struct mmc_host *mmc, struct mmc_request *mrq, > int err) > diff --git a/drivers/mmc/host/sdhci.h b/drivers/mmc/host/sdhci.h > index df7fa0c0ebf8..c2f989dc2361 100644 > --- a/drivers/mmc/host/sdhci.h > +++ b/drivers/mmc/host/sdhci.h > @@ -850,6 +850,8 @@ void sdhci_set_bus_width(struct sdhci_host *host, int width); > void sdhci_reset(struct sdhci_host *host, u8 mask); > void sdhci_set_uhs_signaling(struct sdhci_host *host, unsigned timing); > int sdhci_execute_tuning(struct mmc_host *mmc, u32 opcode); > +void sdhci_enable_preset_value(struct sdhci_host *host, bool enable); > +void sdhci_set_ios_common(struct mmc_host *mmc, struct mmc_ios *ios); > void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios); > int sdhci_start_signal_voltage_switch(struct mmc_host *mmc, > struct mmc_ios *ios);
Hi, Adrian On Fri, Jan 6, 2023 at 5:28 AM Adrian Hunter <adrian.hunter@intel.com> wrote: > > On 13/12/22 11:00, Victor Shih wrote: > > This is a sdhci version of mmc's set_ios operation. > > It covers both UHS-I and UHS-II. > > > > Signed-off-by: Ben Chuang <ben.chuang@genesyslogic.com.tw> > > Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org> > > Signed-off-by: Victor Shih <victor.shih@genesyslogic.com.tw> > > --- > > drivers/mmc/host/sdhci-uhs2.c | 96 +++++++++++++++++++++++++++++++++++ > > drivers/mmc/host/sdhci-uhs2.h | 1 + > > drivers/mmc/host/sdhci.c | 53 +++++++++++-------- > > drivers/mmc/host/sdhci.h | 2 + > > 4 files changed, 131 insertions(+), 21 deletions(-) > > > > diff --git a/drivers/mmc/host/sdhci-uhs2.c b/drivers/mmc/host/sdhci-uhs2.c > > index 00b1b69b49ea..3d52d35a91a5 100644 > > --- a/drivers/mmc/host/sdhci-uhs2.c > > +++ b/drivers/mmc/host/sdhci-uhs2.c > > @@ -213,6 +213,68 @@ void sdhci_uhs2_set_timeout(struct sdhci_host *host, struct mmc_command *cmd) > > } > > EXPORT_SYMBOL_GPL(sdhci_uhs2_set_timeout); > > > > +/** > > + * sdhci_uhs2_clear_set_irqs - set Error Interrupt Status Enable register > > + * @host: SDHCI host > > + * @clear: bit-wise clear mask > > + * @set: bit-wise set mask > > + * > > + * Set/unset bits in UHS-II Error Interrupt Status Enable register > > + */ > > +void sdhci_uhs2_clear_set_irqs(struct sdhci_host *host, u32 clear, u32 set) > > +{ > > + u32 ier; > > + > > + ier = sdhci_readl(host, SDHCI_UHS2_INT_STATUS_ENABLE); > > + ier &= ~clear; > > + ier |= set; > > + sdhci_writel(host, ier, SDHCI_UHS2_INT_STATUS_ENABLE); > > + sdhci_writel(host, ier, SDHCI_UHS2_INT_SIGNAL_ENABLE); > > +} > > +EXPORT_SYMBOL_GPL(sdhci_uhs2_clear_set_irqs); > > + > > +static void __sdhci_uhs2_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > > +{ > > + struct sdhci_host *host = mmc_priv(mmc); > > + u8 cmd_res, dead_lock; > > + u16 ctrl_2; > > + > > + /* UHS2 Timeout Control */ > > + sdhci_calc_timeout_uhs2(host, &cmd_res, &dead_lock); > > + > > + /* change to use calculate value */ > > + cmd_res |= FIELD_PREP(SDHCI_UHS2_TIMER_CTRL_DEADLOCK_MASK, dead_lock); > > + > > + sdhci_uhs2_clear_set_irqs(host, > > + SDHCI_UHS2_INT_CMD_TIMEOUT | > > + SDHCI_UHS2_INT_DEADLOCK_TIMEOUT, > > + 0); > > + sdhci_writeb(host, cmd_res, SDHCI_UHS2_TIMER_CTRL); > > + sdhci_uhs2_clear_set_irqs(host, 0, > > + SDHCI_UHS2_INT_CMD_TIMEOUT | > > + SDHCI_UHS2_INT_DEADLOCK_TIMEOUT); > > + > > + /* UHS2 timing */ > > Please extend comment to include: > > Note, UHS2 timing is disabled when powering off > > > + ctrl_2 = sdhci_readw(host, SDHCI_HOST_CONTROL2); > > + if (ios->timing == MMC_TIMING_SD_UHS2) > > + ctrl_2 |= SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE; > > + else > > + ctrl_2 &= ~(SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE); > > + sdhci_writew(host, ctrl_2, SDHCI_HOST_CONTROL2); > > + > > + if (!(host->quirks2 & SDHCI_QUIRK2_PRESET_VALUE_BROKEN)) > > + sdhci_enable_preset_value(host, true); > > + > > + if (host->ops->set_power) > > + host->ops->set_power(host, ios->power_mode, ios->vdd); > > + else > > + sdhci_uhs2_set_power(host, ios->power_mode, ios->vdd); > > sdhci_set_ios_common() already set the power. Are both needed? > > > + udelay(100); > > Please add a comment for why this delay is here. > > > + > > + host->timing = ios->timing; > > Please move this up to where the timing change is. > The ios->timing was changed in sd_uhs2_power_up function and not modified in this function(__sdhci_uhs2_set_ios), therefore would you want me to move this to the sd_uhs2_power_up function? As I know the ios->timing is used in sdhci_set_clock function, setting the host->timing before call the sdhci_set_clock function ensures that the host->timing used in sdhci_set_clock function is the most recent value. What do you think about this? > > + sdhci_set_clock(host, host->clock); > > +} > > + > > /*****************************************************************************\ > > * * > > * MMC callbacks * > > @@ -234,6 +296,39 @@ static int sdhci_uhs2_start_signal_voltage_switch(struct mmc_host *mmc, > > return sdhci_start_signal_voltage_switch(mmc, ios); > > } > > > > +int sdhci_uhs2_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > > Should be static > > > +{ > > + struct sdhci_host *host = mmc_priv(mmc); > > + > > + if (!(host->version >= SDHCI_SPEC_400) || > > + !(host->mmc->flags & MMC_UHS2_SUPPORT && > > + host->mmc->caps2 & MMC_CAP2_SD_UHS2)) { > > This can be just: > > if (!sdhci_uhs2_mode(host)) { > > Not sure if this is actually possible? > > > + sdhci_set_ios(mmc, ios); > > + return 0; > > + } > > + > > + if (ios->power_mode == MMC_POWER_UNDEFINED) > > + return 0; > > + > > + if (host->flags & SDHCI_DEVICE_DEAD) { > > + if (!IS_ERR(mmc->supply.vmmc) && > > + ios->power_mode == MMC_POWER_OFF) > > + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); > > + if (!IS_ERR_OR_NULL(mmc->supply.vmmc2) && > > + ios->power_mode == MMC_POWER_OFF) > > + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc2, 0); > > This can be just: > > if (ios->power_mode == MMC_POWER_OFF) { > mmc_opt_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); > mmc_opt_regulator_set_ocr(mmc, mmc->supply.vmmc2, 0); > } > > > + return -1; > > + } > > + > > + host->timing = ios->timing; > > __sdhci_uhs2_set_ios() does this so it is not needed here. > > > + > > + sdhci_set_ios_common(mmc, ios); > > + > > + __sdhci_uhs2_set_ios(mmc, ios); > > + > > + return 0; > > +} > > + > > /*****************************************************************************\ > > * * > > * Driver init/exit * > > @@ -244,6 +339,7 @@ static int sdhci_uhs2_host_ops_init(struct sdhci_host *host) > > { > > host->mmc_host_ops.start_signal_voltage_switch = > > sdhci_uhs2_start_signal_voltage_switch; > > + host->mmc_host_ops.uhs2_set_ios = sdhci_uhs2_set_ios; > > > > return 0; > > } > > diff --git a/drivers/mmc/host/sdhci-uhs2.h b/drivers/mmc/host/sdhci-uhs2.h > > index a58ef19c08aa..184fee80253c 100644 > > --- a/drivers/mmc/host/sdhci-uhs2.h > > +++ b/drivers/mmc/host/sdhci-uhs2.h > > @@ -178,5 +178,6 @@ void sdhci_uhs2_dump_regs(struct sdhci_host *host); > > bool sdhci_uhs2_mode(struct sdhci_host *host); > > void sdhci_uhs2_reset(struct sdhci_host *host, u16 mask); > > void sdhci_uhs2_set_timeout(struct sdhci_host *host, struct mmc_command *cmd); > > +void sdhci_uhs2_clear_set_irqs(struct sdhci_host *host, u32 clear, u32 set); > > > > #endif /* __SDHCI_UHS2_H */ > > diff --git a/drivers/mmc/host/sdhci.c b/drivers/mmc/host/sdhci.c > > index 99633a3ef549..49bbdc155b2b 100644 > > --- a/drivers/mmc/host/sdhci.c > > +++ b/drivers/mmc/host/sdhci.c > > @@ -47,8 +47,6 @@ > > static unsigned int debug_quirks = 0; > > static unsigned int debug_quirks2; > > > > -static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable); > > - > > static bool sdhci_send_command(struct sdhci_host *host, struct mmc_command *cmd); > > > > void sdhci_dumpregs(struct sdhci_host *host) > > @@ -1877,6 +1875,9 @@ static u16 sdhci_get_preset_value(struct sdhci_host *host) > > case MMC_TIMING_MMC_HS400: > > preset = sdhci_readw(host, SDHCI_PRESET_FOR_HS400); > > break; > > + case MMC_TIMING_SD_UHS2: > > + preset = sdhci_readw(host, SDHCI_PRESET_FOR_UHS2); > > + break; > > default: > > pr_warn("%s: Invalid UHS-I mode selected\n", > > mmc_hostname(host->mmc)); > > @@ -2325,24 +2326,9 @@ static bool sdhci_presetable_values_change(struct sdhci_host *host, struct mmc_i > > (sdhci_preset_needed(host, ios->timing) || host->drv_type != ios->drv_type); > > } > > > > -void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > > +void sdhci_set_ios_common(struct mmc_host *mmc, struct mmc_ios *ios) > > { > > struct sdhci_host *host = mmc_priv(mmc); > > - bool reinit_uhs = host->reinit_uhs; > > - bool turning_on_clk = false; > > - u8 ctrl; > > - > > - host->reinit_uhs = false; > > - > > - if (ios->power_mode == MMC_POWER_UNDEFINED) > > - return; > > - > > - if (host->flags & SDHCI_DEVICE_DEAD) { > > - if (!IS_ERR(mmc->supply.vmmc) && > > - ios->power_mode == MMC_POWER_OFF) > > - mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); > > - return; > > - } > > > > /* > > * Reset the chip on each power off. > > @@ -2359,8 +2345,6 @@ void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > > sdhci_enable_preset_value(host, false); > > > > if (!ios->clock || ios->clock != host->clock) { > > - turning_on_clk = ios->clock && !host->clock; > > - > > host->ops->set_clock(host, ios->clock); > > host->clock = ios->clock; > > > > @@ -2381,6 +2365,32 @@ void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > > host->ops->set_power(host, ios->power_mode, ios->vdd); > > else > > sdhci_set_power(host, ios->power_mode, ios->vdd); > > +} > > +EXPORT_SYMBOL_GPL(sdhci_set_ios_common); > > + > > +void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > > +{ > > + struct sdhci_host *host = mmc_priv(mmc); > > + bool reinit_uhs = host->reinit_uhs; > > + bool turning_on_clk = false; > > + u8 ctrl; > > + > > + host->reinit_uhs = false; > > + > > + if (ios->power_mode == MMC_POWER_UNDEFINED) > > + return; > > + > > + if (host->flags & SDHCI_DEVICE_DEAD) { > > + if (!IS_ERR(mmc->supply.vmmc) && > > + ios->power_mode == MMC_POWER_OFF) > > + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); > > + return; > > + } > > + > > + sdhci_set_ios_common(mmc, ios); > > + > > + if (!ios->clock || ios->clock != host->clock) > > + turning_on_clk = ios->clock && !host->clock; > > > > if (host->ops->platform_send_init_74_clocks) > > host->ops->platform_send_init_74_clocks(host, ios->power_mode); > > @@ -2959,7 +2969,7 @@ int sdhci_execute_tuning(struct mmc_host *mmc, u32 opcode) > > } > > EXPORT_SYMBOL_GPL(sdhci_execute_tuning); > > > > -static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) > > +void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) > > { > > /* Host Controller v3.00 defines preset value registers */ > > if (host->version < SDHCI_SPEC_300) > > @@ -2987,6 +2997,7 @@ static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) > > host->preset_enabled = enable; > > } > > } > > +EXPORT_SYMBOL_GPL(sdhci_enable_preset_value); > > > > static void sdhci_post_req(struct mmc_host *mmc, struct mmc_request *mrq, > > int err) > > diff --git a/drivers/mmc/host/sdhci.h b/drivers/mmc/host/sdhci.h > > index df7fa0c0ebf8..c2f989dc2361 100644 > > --- a/drivers/mmc/host/sdhci.h > > +++ b/drivers/mmc/host/sdhci.h > > @@ -850,6 +850,8 @@ void sdhci_set_bus_width(struct sdhci_host *host, int width); > > void sdhci_reset(struct sdhci_host *host, u8 mask); > > void sdhci_set_uhs_signaling(struct sdhci_host *host, unsigned timing); > > int sdhci_execute_tuning(struct mmc_host *mmc, u32 opcode); > > +void sdhci_enable_preset_value(struct sdhci_host *host, bool enable); > > +void sdhci_set_ios_common(struct mmc_host *mmc, struct mmc_ios *ios); > > void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios); > > int sdhci_start_signal_voltage_switch(struct mmc_host *mmc, > > struct mmc_ios *ios); > Thanks, Victor Shih
On 26/01/23 12:58, Victor Shih wrote: > Hi, Adrian > > On Fri, Jan 6, 2023 at 5:28 AM Adrian Hunter <adrian.hunter@intel.com> wrote: >> >> On 13/12/22 11:00, Victor Shih wrote: >>> This is a sdhci version of mmc's set_ios operation. >>> It covers both UHS-I and UHS-II. >>> >>> Signed-off-by: Ben Chuang <ben.chuang@genesyslogic.com.tw> >>> Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org> >>> Signed-off-by: Victor Shih <victor.shih@genesyslogic.com.tw> >>> --- >>> drivers/mmc/host/sdhci-uhs2.c | 96 +++++++++++++++++++++++++++++++++++ >>> drivers/mmc/host/sdhci-uhs2.h | 1 + >>> drivers/mmc/host/sdhci.c | 53 +++++++++++-------- >>> drivers/mmc/host/sdhci.h | 2 + >>> 4 files changed, 131 insertions(+), 21 deletions(-) >>> >>> diff --git a/drivers/mmc/host/sdhci-uhs2.c b/drivers/mmc/host/sdhci-uhs2.c >>> index 00b1b69b49ea..3d52d35a91a5 100644 >>> --- a/drivers/mmc/host/sdhci-uhs2.c >>> +++ b/drivers/mmc/host/sdhci-uhs2.c >>> @@ -213,6 +213,68 @@ void sdhci_uhs2_set_timeout(struct sdhci_host *host, struct mmc_command *cmd) >>> } >>> EXPORT_SYMBOL_GPL(sdhci_uhs2_set_timeout); >>> >>> +/** >>> + * sdhci_uhs2_clear_set_irqs - set Error Interrupt Status Enable register >>> + * @host: SDHCI host >>> + * @clear: bit-wise clear mask >>> + * @set: bit-wise set mask >>> + * >>> + * Set/unset bits in UHS-II Error Interrupt Status Enable register >>> + */ >>> +void sdhci_uhs2_clear_set_irqs(struct sdhci_host *host, u32 clear, u32 set) >>> +{ >>> + u32 ier; >>> + >>> + ier = sdhci_readl(host, SDHCI_UHS2_INT_STATUS_ENABLE); >>> + ier &= ~clear; >>> + ier |= set; >>> + sdhci_writel(host, ier, SDHCI_UHS2_INT_STATUS_ENABLE); >>> + sdhci_writel(host, ier, SDHCI_UHS2_INT_SIGNAL_ENABLE); >>> +} >>> +EXPORT_SYMBOL_GPL(sdhci_uhs2_clear_set_irqs); >>> + >>> +static void __sdhci_uhs2_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) >>> +{ >>> + struct sdhci_host *host = mmc_priv(mmc); >>> + u8 cmd_res, dead_lock; >>> + u16 ctrl_2; >>> + >>> + /* UHS2 Timeout Control */ >>> + sdhci_calc_timeout_uhs2(host, &cmd_res, &dead_lock); >>> + >>> + /* change to use calculate value */ >>> + cmd_res |= FIELD_PREP(SDHCI_UHS2_TIMER_CTRL_DEADLOCK_MASK, dead_lock); >>> + >>> + sdhci_uhs2_clear_set_irqs(host, >>> + SDHCI_UHS2_INT_CMD_TIMEOUT | >>> + SDHCI_UHS2_INT_DEADLOCK_TIMEOUT, >>> + 0); >>> + sdhci_writeb(host, cmd_res, SDHCI_UHS2_TIMER_CTRL); >>> + sdhci_uhs2_clear_set_irqs(host, 0, >>> + SDHCI_UHS2_INT_CMD_TIMEOUT | >>> + SDHCI_UHS2_INT_DEADLOCK_TIMEOUT); >>> + >>> + /* UHS2 timing */ >> >> Please extend comment to include: >> >> Note, UHS2 timing is disabled when powering off >> >>> + ctrl_2 = sdhci_readw(host, SDHCI_HOST_CONTROL2); >>> + if (ios->timing == MMC_TIMING_SD_UHS2) >>> + ctrl_2 |= SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE; >>> + else >>> + ctrl_2 &= ~(SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE); >>> + sdhci_writew(host, ctrl_2, SDHCI_HOST_CONTROL2); >>> + >>> + if (!(host->quirks2 & SDHCI_QUIRK2_PRESET_VALUE_BROKEN)) >>> + sdhci_enable_preset_value(host, true); >>> + >>> + if (host->ops->set_power) >>> + host->ops->set_power(host, ios->power_mode, ios->vdd); >>> + else >>> + sdhci_uhs2_set_power(host, ios->power_mode, ios->vdd); >> >> sdhci_set_ios_common() already set the power. Are both needed? >> >>> + udelay(100); >> >> Please add a comment for why this delay is here. >> >>> + >>> + host->timing = ios->timing; >> >> Please move this up to where the timing change is. >> > > The ios->timing was changed in sd_uhs2_power_up function and not > modified in this function(__sdhci_uhs2_set_ios), therefore would you > want me to move this to the sd_uhs2_power_up function? No > > As I know the ios->timing is used in sdhci_set_clock function, setting > the host->timing before call the sdhci_set_clock function ensures that > the host->timing used in sdhci_set_clock function is the most recent > value. What do you think about this? I just meant move it below the timing chunk i.e. /* UHS2 timing. Note, UHS2 timing is disabled when powering off */ ctrl_2 = sdhci_readw(host, SDHCI_HOST_CONTROL2); if (ios->timing == MMC_TIMING_SD_UHS2) ctrl_2 |= SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE; else ctrl_2 &= ~(SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE); sdhci_writew(host, ctrl_2, SDHCI_HOST_CONTROL2); host->timing = ios->timing; > >>> + sdhci_set_clock(host, host->clock); >>> +} >>> + >>> /*****************************************************************************\ >>> * * >>> * MMC callbacks * >>> @@ -234,6 +296,39 @@ static int sdhci_uhs2_start_signal_voltage_switch(struct mmc_host *mmc, >>> return sdhci_start_signal_voltage_switch(mmc, ios); >>> } >>> >>> +int sdhci_uhs2_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) >> >> Should be static >> >>> +{ >>> + struct sdhci_host *host = mmc_priv(mmc); >>> + >>> + if (!(host->version >= SDHCI_SPEC_400) || >>> + !(host->mmc->flags & MMC_UHS2_SUPPORT && >>> + host->mmc->caps2 & MMC_CAP2_SD_UHS2)) { >> >> This can be just: >> >> if (!sdhci_uhs2_mode(host)) { >> >> Not sure if this is actually possible? >> >>> + sdhci_set_ios(mmc, ios); >>> + return 0; >>> + } >>> + >>> + if (ios->power_mode == MMC_POWER_UNDEFINED) >>> + return 0; >>> + >>> + if (host->flags & SDHCI_DEVICE_DEAD) { >>> + if (!IS_ERR(mmc->supply.vmmc) && >>> + ios->power_mode == MMC_POWER_OFF) >>> + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); >>> + if (!IS_ERR_OR_NULL(mmc->supply.vmmc2) && >>> + ios->power_mode == MMC_POWER_OFF) >>> + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc2, 0); >> >> This can be just: >> >> if (ios->power_mode == MMC_POWER_OFF) { >> mmc_opt_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); >> mmc_opt_regulator_set_ocr(mmc, mmc->supply.vmmc2, 0); >> } >> >>> + return -1; >>> + } >>> + >>> + host->timing = ios->timing; >> >> __sdhci_uhs2_set_ios() does this so it is not needed here. >> >>> + >>> + sdhci_set_ios_common(mmc, ios); >>> + >>> + __sdhci_uhs2_set_ios(mmc, ios); >>> + >>> + return 0; >>> +} >>> + >>> /*****************************************************************************\ >>> * * >>> * Driver init/exit * >>> @@ -244,6 +339,7 @@ static int sdhci_uhs2_host_ops_init(struct sdhci_host *host) >>> { >>> host->mmc_host_ops.start_signal_voltage_switch = >>> sdhci_uhs2_start_signal_voltage_switch; >>> + host->mmc_host_ops.uhs2_set_ios = sdhci_uhs2_set_ios; >>> >>> return 0; >>> } >>> diff --git a/drivers/mmc/host/sdhci-uhs2.h b/drivers/mmc/host/sdhci-uhs2.h >>> index a58ef19c08aa..184fee80253c 100644 >>> --- a/drivers/mmc/host/sdhci-uhs2.h >>> +++ b/drivers/mmc/host/sdhci-uhs2.h >>> @@ -178,5 +178,6 @@ void sdhci_uhs2_dump_regs(struct sdhci_host *host); >>> bool sdhci_uhs2_mode(struct sdhci_host *host); >>> void sdhci_uhs2_reset(struct sdhci_host *host, u16 mask); >>> void sdhci_uhs2_set_timeout(struct sdhci_host *host, struct mmc_command *cmd); >>> +void sdhci_uhs2_clear_set_irqs(struct sdhci_host *host, u32 clear, u32 set); >>> >>> #endif /* __SDHCI_UHS2_H */ >>> diff --git a/drivers/mmc/host/sdhci.c b/drivers/mmc/host/sdhci.c >>> index 99633a3ef549..49bbdc155b2b 100644 >>> --- a/drivers/mmc/host/sdhci.c >>> +++ b/drivers/mmc/host/sdhci.c >>> @@ -47,8 +47,6 @@ >>> static unsigned int debug_quirks = 0; >>> static unsigned int debug_quirks2; >>> >>> -static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable); >>> - >>> static bool sdhci_send_command(struct sdhci_host *host, struct mmc_command *cmd); >>> >>> void sdhci_dumpregs(struct sdhci_host *host) >>> @@ -1877,6 +1875,9 @@ static u16 sdhci_get_preset_value(struct sdhci_host *host) >>> case MMC_TIMING_MMC_HS400: >>> preset = sdhci_readw(host, SDHCI_PRESET_FOR_HS400); >>> break; >>> + case MMC_TIMING_SD_UHS2: >>> + preset = sdhci_readw(host, SDHCI_PRESET_FOR_UHS2); >>> + break; >>> default: >>> pr_warn("%s: Invalid UHS-I mode selected\n", >>> mmc_hostname(host->mmc)); >>> @@ -2325,24 +2326,9 @@ static bool sdhci_presetable_values_change(struct sdhci_host *host, struct mmc_i >>> (sdhci_preset_needed(host, ios->timing) || host->drv_type != ios->drv_type); >>> } >>> >>> -void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) >>> +void sdhci_set_ios_common(struct mmc_host *mmc, struct mmc_ios *ios) >>> { >>> struct sdhci_host *host = mmc_priv(mmc); >>> - bool reinit_uhs = host->reinit_uhs; >>> - bool turning_on_clk = false; >>> - u8 ctrl; >>> - >>> - host->reinit_uhs = false; >>> - >>> - if (ios->power_mode == MMC_POWER_UNDEFINED) >>> - return; >>> - >>> - if (host->flags & SDHCI_DEVICE_DEAD) { >>> - if (!IS_ERR(mmc->supply.vmmc) && >>> - ios->power_mode == MMC_POWER_OFF) >>> - mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); >>> - return; >>> - } >>> >>> /* >>> * Reset the chip on each power off. >>> @@ -2359,8 +2345,6 @@ void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) >>> sdhci_enable_preset_value(host, false); >>> >>> if (!ios->clock || ios->clock != host->clock) { >>> - turning_on_clk = ios->clock && !host->clock; >>> - >>> host->ops->set_clock(host, ios->clock); >>> host->clock = ios->clock; >>> >>> @@ -2381,6 +2365,32 @@ void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) >>> host->ops->set_power(host, ios->power_mode, ios->vdd); >>> else >>> sdhci_set_power(host, ios->power_mode, ios->vdd); >>> +} >>> +EXPORT_SYMBOL_GPL(sdhci_set_ios_common); >>> + >>> +void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) >>> +{ >>> + struct sdhci_host *host = mmc_priv(mmc); >>> + bool reinit_uhs = host->reinit_uhs; >>> + bool turning_on_clk = false; >>> + u8 ctrl; >>> + >>> + host->reinit_uhs = false; >>> + >>> + if (ios->power_mode == MMC_POWER_UNDEFINED) >>> + return; >>> + >>> + if (host->flags & SDHCI_DEVICE_DEAD) { >>> + if (!IS_ERR(mmc->supply.vmmc) && >>> + ios->power_mode == MMC_POWER_OFF) >>> + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); >>> + return; >>> + } >>> + >>> + sdhci_set_ios_common(mmc, ios); >>> + >>> + if (!ios->clock || ios->clock != host->clock) >>> + turning_on_clk = ios->clock && !host->clock; >>> >>> if (host->ops->platform_send_init_74_clocks) >>> host->ops->platform_send_init_74_clocks(host, ios->power_mode); >>> @@ -2959,7 +2969,7 @@ int sdhci_execute_tuning(struct mmc_host *mmc, u32 opcode) >>> } >>> EXPORT_SYMBOL_GPL(sdhci_execute_tuning); >>> >>> -static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) >>> +void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) >>> { >>> /* Host Controller v3.00 defines preset value registers */ >>> if (host->version < SDHCI_SPEC_300) >>> @@ -2987,6 +2997,7 @@ static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) >>> host->preset_enabled = enable; >>> } >>> } >>> +EXPORT_SYMBOL_GPL(sdhci_enable_preset_value); >>> >>> static void sdhci_post_req(struct mmc_host *mmc, struct mmc_request *mrq, >>> int err) >>> diff --git a/drivers/mmc/host/sdhci.h b/drivers/mmc/host/sdhci.h >>> index df7fa0c0ebf8..c2f989dc2361 100644 >>> --- a/drivers/mmc/host/sdhci.h >>> +++ b/drivers/mmc/host/sdhci.h >>> @@ -850,6 +850,8 @@ void sdhci_set_bus_width(struct sdhci_host *host, int width); >>> void sdhci_reset(struct sdhci_host *host, u8 mask); >>> void sdhci_set_uhs_signaling(struct sdhci_host *host, unsigned timing); >>> int sdhci_execute_tuning(struct mmc_host *mmc, u32 opcode); >>> +void sdhci_enable_preset_value(struct sdhci_host *host, bool enable); >>> +void sdhci_set_ios_common(struct mmc_host *mmc, struct mmc_ios *ios); >>> void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios); >>> int sdhci_start_signal_voltage_switch(struct mmc_host *mmc, >>> struct mmc_ios *ios); >> > > Thanks, Victor Shih
Hi, Adrian On Mon, Jan 30, 2023 at 9:16 PM Adrian Hunter <adrian.hunter@intel.com> wrote: > > On 26/01/23 12:58, Victor Shih wrote: > > Hi, Adrian > > > > On Fri, Jan 6, 2023 at 5:28 AM Adrian Hunter <adrian.hunter@intel.com> wrote: > >> > >> On 13/12/22 11:00, Victor Shih wrote: > >>> This is a sdhci version of mmc's set_ios operation. > >>> It covers both UHS-I and UHS-II. > >>> > >>> Signed-off-by: Ben Chuang <ben.chuang@genesyslogic.com.tw> > >>> Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org> > >>> Signed-off-by: Victor Shih <victor.shih@genesyslogic.com.tw> > >>> --- > >>> drivers/mmc/host/sdhci-uhs2.c | 96 +++++++++++++++++++++++++++++++++++ > >>> drivers/mmc/host/sdhci-uhs2.h | 1 + > >>> drivers/mmc/host/sdhci.c | 53 +++++++++++-------- > >>> drivers/mmc/host/sdhci.h | 2 + > >>> 4 files changed, 131 insertions(+), 21 deletions(-) > >>> > >>> diff --git a/drivers/mmc/host/sdhci-uhs2.c b/drivers/mmc/host/sdhci-uhs2.c > >>> index 00b1b69b49ea..3d52d35a91a5 100644 > >>> --- a/drivers/mmc/host/sdhci-uhs2.c > >>> +++ b/drivers/mmc/host/sdhci-uhs2.c > >>> @@ -213,6 +213,68 @@ void sdhci_uhs2_set_timeout(struct sdhci_host *host, struct mmc_command *cmd) > >>> } > >>> EXPORT_SYMBOL_GPL(sdhci_uhs2_set_timeout); > >>> > >>> +/** > >>> + * sdhci_uhs2_clear_set_irqs - set Error Interrupt Status Enable register > >>> + * @host: SDHCI host > >>> + * @clear: bit-wise clear mask > >>> + * @set: bit-wise set mask > >>> + * > >>> + * Set/unset bits in UHS-II Error Interrupt Status Enable register > >>> + */ > >>> +void sdhci_uhs2_clear_set_irqs(struct sdhci_host *host, u32 clear, u32 set) > >>> +{ > >>> + u32 ier; > >>> + > >>> + ier = sdhci_readl(host, SDHCI_UHS2_INT_STATUS_ENABLE); > >>> + ier &= ~clear; > >>> + ier |= set; > >>> + sdhci_writel(host, ier, SDHCI_UHS2_INT_STATUS_ENABLE); > >>> + sdhci_writel(host, ier, SDHCI_UHS2_INT_SIGNAL_ENABLE); > >>> +} > >>> +EXPORT_SYMBOL_GPL(sdhci_uhs2_clear_set_irqs); > >>> + > >>> +static void __sdhci_uhs2_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > >>> +{ > >>> + struct sdhci_host *host = mmc_priv(mmc); > >>> + u8 cmd_res, dead_lock; > >>> + u16 ctrl_2; > >>> + > >>> + /* UHS2 Timeout Control */ > >>> + sdhci_calc_timeout_uhs2(host, &cmd_res, &dead_lock); > >>> + > >>> + /* change to use calculate value */ > >>> + cmd_res |= FIELD_PREP(SDHCI_UHS2_TIMER_CTRL_DEADLOCK_MASK, dead_lock); > >>> + > >>> + sdhci_uhs2_clear_set_irqs(host, > >>> + SDHCI_UHS2_INT_CMD_TIMEOUT | > >>> + SDHCI_UHS2_INT_DEADLOCK_TIMEOUT, > >>> + 0); > >>> + sdhci_writeb(host, cmd_res, SDHCI_UHS2_TIMER_CTRL); > >>> + sdhci_uhs2_clear_set_irqs(host, 0, > >>> + SDHCI_UHS2_INT_CMD_TIMEOUT | > >>> + SDHCI_UHS2_INT_DEADLOCK_TIMEOUT); > >>> + > >>> + /* UHS2 timing */ > >> > >> Please extend comment to include: > >> > >> Note, UHS2 timing is disabled when powering off > >> I will update it in V7 version. > >>> + ctrl_2 = sdhci_readw(host, SDHCI_HOST_CONTROL2); > >>> + if (ios->timing == MMC_TIMING_SD_UHS2) > >>> + ctrl_2 |= SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE; > >>> + else > >>> + ctrl_2 &= ~(SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE); > >>> + sdhci_writew(host, ctrl_2, SDHCI_HOST_CONTROL2); > >>> + > >>> + if (!(host->quirks2 & SDHCI_QUIRK2_PRESET_VALUE_BROKEN)) > >>> + sdhci_enable_preset_value(host, true); > >>> + > >>> + if (host->ops->set_power) > >>> + host->ops->set_power(host, ios->power_mode, ios->vdd); > >>> + else > >>> + sdhci_uhs2_set_power(host, ios->power_mode, ios->vdd); > >> > >> sdhci_set_ios_common() already set the power. Are both needed? > >> I will move the set_power function in the sdhci_set_ios_common to the sdhci_set_ios and update it in V7 version. > >>> + udelay(100); > >> > >> Please add a comment for why this delay is here. > >> I will drop it and update in V7 version. > >>> + > >>> + host->timing = ios->timing; > >> > >> Please move this up to where the timing change is. > >> > > > > The ios->timing was changed in sd_uhs2_power_up function and not > > modified in this function(__sdhci_uhs2_set_ios), therefore would you > > want me to move this to the sd_uhs2_power_up function? > > No > > > > > As I know the ios->timing is used in sdhci_set_clock function, setting > > the host->timing before call the sdhci_set_clock function ensures that > > the host->timing used in sdhci_set_clock function is the most recent > > value. What do you think about this? > > I just meant move it below the timing chunk i.e. > > /* UHS2 timing. Note, UHS2 timing is disabled when powering off */ > ctrl_2 = sdhci_readw(host, SDHCI_HOST_CONTROL2); > if (ios->timing == MMC_TIMING_SD_UHS2) > ctrl_2 |= SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE; > else > ctrl_2 &= ~(SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE); > sdhci_writew(host, ctrl_2, SDHCI_HOST_CONTROL2); > host->timing = ios->timing; > I will update it in V7 version. > > > >>> + sdhci_set_clock(host, host->clock); > >>> +} > >>> + > >>> /*****************************************************************************\ > >>> * * > >>> * MMC callbacks * > >>> @@ -234,6 +296,39 @@ static int sdhci_uhs2_start_signal_voltage_switch(struct mmc_host *mmc, > >>> return sdhci_start_signal_voltage_switch(mmc, ios); > >>> } > >>> > >>> +int sdhci_uhs2_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > >> > >> Should be static > >> I will update it in V7 version. > >>> +{ > >>> + struct sdhci_host *host = mmc_priv(mmc); > >>> + > >>> + if (!(host->version >= SDHCI_SPEC_400) || > >>> + !(host->mmc->flags & MMC_UHS2_SUPPORT && > >>> + host->mmc->caps2 & MMC_CAP2_SD_UHS2)) { > >> > >> This can be just: > >> > >> if (!sdhci_uhs2_mode(host)) { > >> > >> Not sure if this is actually possible? > >> I will update it in V7 version. > >>> + sdhci_set_ios(mmc, ios); > >>> + return 0; > >>> + } > >>> + > >>> + if (ios->power_mode == MMC_POWER_UNDEFINED) > >>> + return 0; > >>> + > >>> + if (host->flags & SDHCI_DEVICE_DEAD) { > >>> + if (!IS_ERR(mmc->supply.vmmc) && > >>> + ios->power_mode == MMC_POWER_OFF) > >>> + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); > >>> + if (!IS_ERR_OR_NULL(mmc->supply.vmmc2) && > >>> + ios->power_mode == MMC_POWER_OFF) > >>> + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc2, 0); > >> > >> This can be just: > >> > >> if (ios->power_mode == MMC_POWER_OFF) { > >> mmc_opt_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); > >> mmc_opt_regulator_set_ocr(mmc, mmc->supply.vmmc2, 0); > >> } > >> I will update it in V7 version. > >>> + return -1; > >>> + } > >>> + > >>> + host->timing = ios->timing; > >> > >> __sdhci_uhs2_set_ios() does this so it is not needed here. > >> I will update it in V7 version. > >>> + > >>> + sdhci_set_ios_common(mmc, ios); > >>> + > >>> + __sdhci_uhs2_set_ios(mmc, ios); > >>> + > >>> + return 0; > >>> +} > >>> + > >>> /*****************************************************************************\ > >>> * * > >>> * Driver init/exit * > >>> @@ -244,6 +339,7 @@ static int sdhci_uhs2_host_ops_init(struct sdhci_host *host) > >>> { > >>> host->mmc_host_ops.start_signal_voltage_switch = > >>> sdhci_uhs2_start_signal_voltage_switch; > >>> + host->mmc_host_ops.uhs2_set_ios = sdhci_uhs2_set_ios; > >>> > >>> return 0; > >>> } > >>> diff --git a/drivers/mmc/host/sdhci-uhs2.h b/drivers/mmc/host/sdhci-uhs2.h > >>> index a58ef19c08aa..184fee80253c 100644 > >>> --- a/drivers/mmc/host/sdhci-uhs2.h > >>> +++ b/drivers/mmc/host/sdhci-uhs2.h > >>> @@ -178,5 +178,6 @@ void sdhci_uhs2_dump_regs(struct sdhci_host *host); > >>> bool sdhci_uhs2_mode(struct sdhci_host *host); > >>> void sdhci_uhs2_reset(struct sdhci_host *host, u16 mask); > >>> void sdhci_uhs2_set_timeout(struct sdhci_host *host, struct mmc_command *cmd); > >>> +void sdhci_uhs2_clear_set_irqs(struct sdhci_host *host, u32 clear, u32 set); > >>> > >>> #endif /* __SDHCI_UHS2_H */ > >>> diff --git a/drivers/mmc/host/sdhci.c b/drivers/mmc/host/sdhci.c > >>> index 99633a3ef549..49bbdc155b2b 100644 > >>> --- a/drivers/mmc/host/sdhci.c > >>> +++ b/drivers/mmc/host/sdhci.c > >>> @@ -47,8 +47,6 @@ > >>> static unsigned int debug_quirks = 0; > >>> static unsigned int debug_quirks2; > >>> > >>> -static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable); > >>> - > >>> static bool sdhci_send_command(struct sdhci_host *host, struct mmc_command *cmd); > >>> > >>> void sdhci_dumpregs(struct sdhci_host *host) > >>> @@ -1877,6 +1875,9 @@ static u16 sdhci_get_preset_value(struct sdhci_host *host) > >>> case MMC_TIMING_MMC_HS400: > >>> preset = sdhci_readw(host, SDHCI_PRESET_FOR_HS400); > >>> break; > >>> + case MMC_TIMING_SD_UHS2: > >>> + preset = sdhci_readw(host, SDHCI_PRESET_FOR_UHS2); > >>> + break; > >>> default: > >>> pr_warn("%s: Invalid UHS-I mode selected\n", > >>> mmc_hostname(host->mmc)); > >>> @@ -2325,24 +2326,9 @@ static bool sdhci_presetable_values_change(struct sdhci_host *host, struct mmc_i > >>> (sdhci_preset_needed(host, ios->timing) || host->drv_type != ios->drv_type); > >>> } > >>> > >>> -void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > >>> +void sdhci_set_ios_common(struct mmc_host *mmc, struct mmc_ios *ios) > >>> { > >>> struct sdhci_host *host = mmc_priv(mmc); > >>> - bool reinit_uhs = host->reinit_uhs; > >>> - bool turning_on_clk = false; > >>> - u8 ctrl; > >>> - > >>> - host->reinit_uhs = false; > >>> - > >>> - if (ios->power_mode == MMC_POWER_UNDEFINED) > >>> - return; > >>> - > >>> - if (host->flags & SDHCI_DEVICE_DEAD) { > >>> - if (!IS_ERR(mmc->supply.vmmc) && > >>> - ios->power_mode == MMC_POWER_OFF) > >>> - mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); > >>> - return; > >>> - } > >>> > >>> /* > >>> * Reset the chip on each power off. > >>> @@ -2359,8 +2345,6 @@ void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > >>> sdhci_enable_preset_value(host, false); > >>> > >>> if (!ios->clock || ios->clock != host->clock) { > >>> - turning_on_clk = ios->clock && !host->clock; > >>> - > >>> host->ops->set_clock(host, ios->clock); > >>> host->clock = ios->clock; > >>> > >>> @@ -2381,6 +2365,32 @@ void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > >>> host->ops->set_power(host, ios->power_mode, ios->vdd); > >>> else > >>> sdhci_set_power(host, ios->power_mode, ios->vdd); > >>> +} > >>> +EXPORT_SYMBOL_GPL(sdhci_set_ios_common); > >>> + > >>> +void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) > >>> +{ > >>> + struct sdhci_host *host = mmc_priv(mmc); > >>> + bool reinit_uhs = host->reinit_uhs; > >>> + bool turning_on_clk = false; > >>> + u8 ctrl; > >>> + > >>> + host->reinit_uhs = false; > >>> + > >>> + if (ios->power_mode == MMC_POWER_UNDEFINED) > >>> + return; > >>> + > >>> + if (host->flags & SDHCI_DEVICE_DEAD) { > >>> + if (!IS_ERR(mmc->supply.vmmc) && > >>> + ios->power_mode == MMC_POWER_OFF) > >>> + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); > >>> + return; > >>> + } > >>> + > >>> + sdhci_set_ios_common(mmc, ios); > >>> + > >>> + if (!ios->clock || ios->clock != host->clock) > >>> + turning_on_clk = ios->clock && !host->clock; > >>> > >>> if (host->ops->platform_send_init_74_clocks) > >>> host->ops->platform_send_init_74_clocks(host, ios->power_mode); > >>> @@ -2959,7 +2969,7 @@ int sdhci_execute_tuning(struct mmc_host *mmc, u32 opcode) > >>> } > >>> EXPORT_SYMBOL_GPL(sdhci_execute_tuning); > >>> > >>> -static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) > >>> +void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) > >>> { > >>> /* Host Controller v3.00 defines preset value registers */ > >>> if (host->version < SDHCI_SPEC_300) > >>> @@ -2987,6 +2997,7 @@ static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) > >>> host->preset_enabled = enable; > >>> } > >>> } > >>> +EXPORT_SYMBOL_GPL(sdhci_enable_preset_value); > >>> > >>> static void sdhci_post_req(struct mmc_host *mmc, struct mmc_request *mrq, > >>> int err) > >>> diff --git a/drivers/mmc/host/sdhci.h b/drivers/mmc/host/sdhci.h > >>> index df7fa0c0ebf8..c2f989dc2361 100644 > >>> --- a/drivers/mmc/host/sdhci.h > >>> +++ b/drivers/mmc/host/sdhci.h > >>> @@ -850,6 +850,8 @@ void sdhci_set_bus_width(struct sdhci_host *host, int width); > >>> void sdhci_reset(struct sdhci_host *host, u8 mask); > >>> void sdhci_set_uhs_signaling(struct sdhci_host *host, unsigned timing); > >>> int sdhci_execute_tuning(struct mmc_host *mmc, u32 opcode); > >>> +void sdhci_enable_preset_value(struct sdhci_host *host, bool enable); > >>> +void sdhci_set_ios_common(struct mmc_host *mmc, struct mmc_ios *ios); > >>> void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios); > >>> int sdhci_start_signal_voltage_switch(struct mmc_host *mmc, > >>> struct mmc_ios *ios); > >> > > > > Thanks, Victor Shih > Thanks, Victor Shih
diff --git a/drivers/mmc/host/sdhci-uhs2.c b/drivers/mmc/host/sdhci-uhs2.c index 00b1b69b49ea..3d52d35a91a5 100644 --- a/drivers/mmc/host/sdhci-uhs2.c +++ b/drivers/mmc/host/sdhci-uhs2.c @@ -213,6 +213,68 @@ void sdhci_uhs2_set_timeout(struct sdhci_host *host, struct mmc_command *cmd) } EXPORT_SYMBOL_GPL(sdhci_uhs2_set_timeout); +/** + * sdhci_uhs2_clear_set_irqs - set Error Interrupt Status Enable register + * @host: SDHCI host + * @clear: bit-wise clear mask + * @set: bit-wise set mask + * + * Set/unset bits in UHS-II Error Interrupt Status Enable register + */ +void sdhci_uhs2_clear_set_irqs(struct sdhci_host *host, u32 clear, u32 set) +{ + u32 ier; + + ier = sdhci_readl(host, SDHCI_UHS2_INT_STATUS_ENABLE); + ier &= ~clear; + ier |= set; + sdhci_writel(host, ier, SDHCI_UHS2_INT_STATUS_ENABLE); + sdhci_writel(host, ier, SDHCI_UHS2_INT_SIGNAL_ENABLE); +} +EXPORT_SYMBOL_GPL(sdhci_uhs2_clear_set_irqs); + +static void __sdhci_uhs2_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) +{ + struct sdhci_host *host = mmc_priv(mmc); + u8 cmd_res, dead_lock; + u16 ctrl_2; + + /* UHS2 Timeout Control */ + sdhci_calc_timeout_uhs2(host, &cmd_res, &dead_lock); + + /* change to use calculate value */ + cmd_res |= FIELD_PREP(SDHCI_UHS2_TIMER_CTRL_DEADLOCK_MASK, dead_lock); + + sdhci_uhs2_clear_set_irqs(host, + SDHCI_UHS2_INT_CMD_TIMEOUT | + SDHCI_UHS2_INT_DEADLOCK_TIMEOUT, + 0); + sdhci_writeb(host, cmd_res, SDHCI_UHS2_TIMER_CTRL); + sdhci_uhs2_clear_set_irqs(host, 0, + SDHCI_UHS2_INT_CMD_TIMEOUT | + SDHCI_UHS2_INT_DEADLOCK_TIMEOUT); + + /* UHS2 timing */ + ctrl_2 = sdhci_readw(host, SDHCI_HOST_CONTROL2); + if (ios->timing == MMC_TIMING_SD_UHS2) + ctrl_2 |= SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE; + else + ctrl_2 &= ~(SDHCI_CTRL_UHS2 | SDHCI_CTRL_UHS2_ENABLE); + sdhci_writew(host, ctrl_2, SDHCI_HOST_CONTROL2); + + if (!(host->quirks2 & SDHCI_QUIRK2_PRESET_VALUE_BROKEN)) + sdhci_enable_preset_value(host, true); + + if (host->ops->set_power) + host->ops->set_power(host, ios->power_mode, ios->vdd); + else + sdhci_uhs2_set_power(host, ios->power_mode, ios->vdd); + udelay(100); + + host->timing = ios->timing; + sdhci_set_clock(host, host->clock); +} + /*****************************************************************************\ * * * MMC callbacks * @@ -234,6 +296,39 @@ static int sdhci_uhs2_start_signal_voltage_switch(struct mmc_host *mmc, return sdhci_start_signal_voltage_switch(mmc, ios); } +int sdhci_uhs2_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) +{ + struct sdhci_host *host = mmc_priv(mmc); + + if (!(host->version >= SDHCI_SPEC_400) || + !(host->mmc->flags & MMC_UHS2_SUPPORT && + host->mmc->caps2 & MMC_CAP2_SD_UHS2)) { + sdhci_set_ios(mmc, ios); + return 0; + } + + if (ios->power_mode == MMC_POWER_UNDEFINED) + return 0; + + if (host->flags & SDHCI_DEVICE_DEAD) { + if (!IS_ERR(mmc->supply.vmmc) && + ios->power_mode == MMC_POWER_OFF) + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); + if (!IS_ERR_OR_NULL(mmc->supply.vmmc2) && + ios->power_mode == MMC_POWER_OFF) + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc2, 0); + return -1; + } + + host->timing = ios->timing; + + sdhci_set_ios_common(mmc, ios); + + __sdhci_uhs2_set_ios(mmc, ios); + + return 0; +} + /*****************************************************************************\ * * * Driver init/exit * @@ -244,6 +339,7 @@ static int sdhci_uhs2_host_ops_init(struct sdhci_host *host) { host->mmc_host_ops.start_signal_voltage_switch = sdhci_uhs2_start_signal_voltage_switch; + host->mmc_host_ops.uhs2_set_ios = sdhci_uhs2_set_ios; return 0; } diff --git a/drivers/mmc/host/sdhci-uhs2.h b/drivers/mmc/host/sdhci-uhs2.h index a58ef19c08aa..184fee80253c 100644 --- a/drivers/mmc/host/sdhci-uhs2.h +++ b/drivers/mmc/host/sdhci-uhs2.h @@ -178,5 +178,6 @@ void sdhci_uhs2_dump_regs(struct sdhci_host *host); bool sdhci_uhs2_mode(struct sdhci_host *host); void sdhci_uhs2_reset(struct sdhci_host *host, u16 mask); void sdhci_uhs2_set_timeout(struct sdhci_host *host, struct mmc_command *cmd); +void sdhci_uhs2_clear_set_irqs(struct sdhci_host *host, u32 clear, u32 set); #endif /* __SDHCI_UHS2_H */ diff --git a/drivers/mmc/host/sdhci.c b/drivers/mmc/host/sdhci.c index 99633a3ef549..49bbdc155b2b 100644 --- a/drivers/mmc/host/sdhci.c +++ b/drivers/mmc/host/sdhci.c @@ -47,8 +47,6 @@ static unsigned int debug_quirks = 0; static unsigned int debug_quirks2; -static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable); - static bool sdhci_send_command(struct sdhci_host *host, struct mmc_command *cmd); void sdhci_dumpregs(struct sdhci_host *host) @@ -1877,6 +1875,9 @@ static u16 sdhci_get_preset_value(struct sdhci_host *host) case MMC_TIMING_MMC_HS400: preset = sdhci_readw(host, SDHCI_PRESET_FOR_HS400); break; + case MMC_TIMING_SD_UHS2: + preset = sdhci_readw(host, SDHCI_PRESET_FOR_UHS2); + break; default: pr_warn("%s: Invalid UHS-I mode selected\n", mmc_hostname(host->mmc)); @@ -2325,24 +2326,9 @@ static bool sdhci_presetable_values_change(struct sdhci_host *host, struct mmc_i (sdhci_preset_needed(host, ios->timing) || host->drv_type != ios->drv_type); } -void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) +void sdhci_set_ios_common(struct mmc_host *mmc, struct mmc_ios *ios) { struct sdhci_host *host = mmc_priv(mmc); - bool reinit_uhs = host->reinit_uhs; - bool turning_on_clk = false; - u8 ctrl; - - host->reinit_uhs = false; - - if (ios->power_mode == MMC_POWER_UNDEFINED) - return; - - if (host->flags & SDHCI_DEVICE_DEAD) { - if (!IS_ERR(mmc->supply.vmmc) && - ios->power_mode == MMC_POWER_OFF) - mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); - return; - } /* * Reset the chip on each power off. @@ -2359,8 +2345,6 @@ void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) sdhci_enable_preset_value(host, false); if (!ios->clock || ios->clock != host->clock) { - turning_on_clk = ios->clock && !host->clock; - host->ops->set_clock(host, ios->clock); host->clock = ios->clock; @@ -2381,6 +2365,32 @@ void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) host->ops->set_power(host, ios->power_mode, ios->vdd); else sdhci_set_power(host, ios->power_mode, ios->vdd); +} +EXPORT_SYMBOL_GPL(sdhci_set_ios_common); + +void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) +{ + struct sdhci_host *host = mmc_priv(mmc); + bool reinit_uhs = host->reinit_uhs; + bool turning_on_clk = false; + u8 ctrl; + + host->reinit_uhs = false; + + if (ios->power_mode == MMC_POWER_UNDEFINED) + return; + + if (host->flags & SDHCI_DEVICE_DEAD) { + if (!IS_ERR(mmc->supply.vmmc) && + ios->power_mode == MMC_POWER_OFF) + mmc_regulator_set_ocr(mmc, mmc->supply.vmmc, 0); + return; + } + + sdhci_set_ios_common(mmc, ios); + + if (!ios->clock || ios->clock != host->clock) + turning_on_clk = ios->clock && !host->clock; if (host->ops->platform_send_init_74_clocks) host->ops->platform_send_init_74_clocks(host, ios->power_mode); @@ -2959,7 +2969,7 @@ int sdhci_execute_tuning(struct mmc_host *mmc, u32 opcode) } EXPORT_SYMBOL_GPL(sdhci_execute_tuning); -static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) +void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) { /* Host Controller v3.00 defines preset value registers */ if (host->version < SDHCI_SPEC_300) @@ -2987,6 +2997,7 @@ static void sdhci_enable_preset_value(struct sdhci_host *host, bool enable) host->preset_enabled = enable; } } +EXPORT_SYMBOL_GPL(sdhci_enable_preset_value); static void sdhci_post_req(struct mmc_host *mmc, struct mmc_request *mrq, int err) diff --git a/drivers/mmc/host/sdhci.h b/drivers/mmc/host/sdhci.h index df7fa0c0ebf8..c2f989dc2361 100644 --- a/drivers/mmc/host/sdhci.h +++ b/drivers/mmc/host/sdhci.h @@ -850,6 +850,8 @@ void sdhci_set_bus_width(struct sdhci_host *host, int width); void sdhci_reset(struct sdhci_host *host, u8 mask); void sdhci_set_uhs_signaling(struct sdhci_host *host, unsigned timing); int sdhci_execute_tuning(struct mmc_host *mmc, u32 opcode); +void sdhci_enable_preset_value(struct sdhci_host *host, bool enable); +void sdhci_set_ios_common(struct mmc_host *mmc, struct mmc_ios *ios); void sdhci_set_ios(struct mmc_host *mmc, struct mmc_ios *ios); int sdhci_start_signal_voltage_switch(struct mmc_host *mmc, struct mmc_ios *ios);