From mboxrd@z Thu Jan 1 00:00:00 1970 Received: from smtp.gentoo.org (woodpecker.gentoo.org [140.211.166.183]) (using TLSv1.2 with cipher ECDHE-RSA-AES128-GCM-SHA256 (128/128 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 00E1625557 for ; Wed, 22 Oct 2025 00:14:31 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=140.211.166.183 ARC-Seal:i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1761092074; cv=none; b=IlUKBxet50keV+rAO31paNjimbudUq65wcs1iodtyvkHiyvkg0Q69VZ5di4dVbbIPPCtfnuy5mRieWPXa0/OhcFYzto9rbbUsBY+h5gkO9UagJsW6Dsbi8gMD6v4/zOphS/tLuDy+yP5arLXhoKCbrg2d+kEwYoX4XMb2EbMULE= ARC-Message-Signature:i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1761092074; c=relaxed/simple; bh=ebu16OWMkG7uaAXBv6x6mSyjDS0vZjqdXh8uAxXnm6U=; h=Date:From:To:Cc:Subject:Message-ID:References:MIME-Version: Content-Type:Content-Disposition:In-Reply-To; b=qWK+X8J18ib/t+idiGleh2vCouRMeZishzLoMyjkk3uoNAmSM1HP614AcSL+gaeZBd6MyWfyWqsDrflvyX1QFgs4ueMoDpRtdiN+GLYrLP6Ptjbuetjx75V9YPu7XvY8kVg0U6nqGEBVgiQkg3vwdWRV1ORkNDrXxz2SYKGJftA= ARC-Authentication-Results:i=1; smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=gentoo.org; spf=pass smtp.mailfrom=gentoo.org; arc=none smtp.client-ip=140.211.166.183 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=gentoo.org Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=gentoo.org Received: from localhost (unknown [116.232.147.23]) (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits) key-exchange secp256r1 server-signature RSA-PSS (4096 bits) server-digest SHA256) (No client certificate requested) (Authenticated sender: dlan) by smtp.gentoo.org (Postfix) with ESMTPSA id F0261340E9F; Wed, 22 Oct 2025 00:14:30 +0000 (UTC) Date: Wed, 22 Oct 2025 08:14:20 +0800 From: Yixun Lan To: Andre Przywara Cc: Lee Jones , Chen-Yu Tsai , Liam Girdwood , Mark Brown , Samuel Holland , Jernej Skrabec , Rob Herring , Krzysztof Kozlowski , Conor Dooley , devicetree@vger.kernel.org, linux-sunxi@lists.linux.dev, linux-kernel@vger.kernel.org Subject: Re: [PATCH 3/3] regulator: axp20x: add support for the AXP318W Message-ID: <20251022001420-GYA1522542@gentoo.org> References: <20251021112013.2710903-1-andre.przywara@arm.com> <20251021112013.2710903-4-andre.przywara@arm.com> Precedence: bulk X-Mailing-List: linux-sunxi@lists.linux.dev List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: <20251021112013.2710903-4-andre.przywara@arm.com> Hi Andre, On 12:20 Tue 21 Oct , Andre Przywara wrote: > The X-Powers AXP318W is a typical PMIC from X-Powers, featuring nine > DC/DC converters and 28 LDOs, on the regulator side. > > Describe the chip's voltage settings and switch registers, how the > voltages are encoded, and connect this to the MFD device via its > regulator ID. > We use just "318" for the internal identifiers, for easier typing and > less churn. If something else other than the "AXP318W" shows up, that's > an easy change, externally visible strings carry the additional letter > already. > > Signed-off-by: Andre Przywara > --- > drivers/regulator/axp20x-regulator.c | 170 ++++++++++++++++++++++++++- > include/linux/mfd/axp20x.h | 43 +++++++ > 2 files changed, 211 insertions(+), 2 deletions(-) > > diff --git a/drivers/regulator/axp20x-regulator.c b/drivers/regulator/axp20x-regulator.c > index da891415efc0b..1576bf4178f8f 100644 > --- a/drivers/regulator/axp20x-regulator.c > +++ b/drivers/regulator/axp20x-regulator.c > @@ -138,6 +138,15 @@ > #define AXP313A_DCDC_V_OUT_MASK GENMASK(6, 0) > #define AXP313A_LDO_V_OUT_MASK GENMASK(4, 0) > > +#define AXP318_DCDC1_V_OUT_MASK GENMASK(4, 0) > +#define AXP318_DCDC2_V_OUT_MASK GENMASK(6, 0) > +#define AXP318_LDO_V_OUT_MASK GENMASK(4, 0) > +#define AXP318_ELDO_V_OUT_MASK GENMASK(5, 0) > +#define AXP318_DCDC2_NUM_VOLTAGES 88 > +#define AXP318_DCDC6_NUM_VOLTAGES 128 > +#define AXP318_DCDC7_NUM_VOLTAGES 103 > +#define AXP318_DCDC8_NUM_VOLTAGES 119 > + > #define AXP717_DCDC1_NUM_VOLTAGES 88 > #define AXP717_DCDC2_NUM_VOLTAGES 107 > #define AXP717_DCDC3_NUM_VOLTAGES 103 > @@ -765,6 +774,155 @@ static const struct regulator_desc axp313a_regulators[] = { > AXP_DESC_FIXED(AXP313A, RTC_LDO, "rtc-ldo", "vin1", 1800), > }; > > +static const struct linear_range axp318_dcdc2_ranges[] = { > + REGULATOR_LINEAR_RANGE(500000, 0, 70, 10000), > + REGULATOR_LINEAR_RANGE(1220000, 71, 87, 20000), > +}; > + .. > +static const struct linear_range axp318_dcdc6_ranges[] = { > + REGULATOR_LINEAR_RANGE(500000, 0, 70, 10000), > + REGULATOR_LINEAR_RANGE(1220000, 71, 87, 20000), > + REGULATOR_LINEAR_RANGE(1800000, 88, 118, 20000), > + REGULATOR_LINEAR_RANGE(2440000, 119, 127, 40000), > +}; > + > +static const struct linear_range axp318_dcdc7_ranges[] = { > + REGULATOR_LINEAR_RANGE(500000, 0, 70, 10000), > + REGULATOR_LINEAR_RANGE(1220000, 71, 102, 20000), > +}; > + > +static const struct linear_range axp318_dcdc8_ranges[] = { > + REGULATOR_LINEAR_RANGE(500000, 0, 70, 10000), > + REGULATOR_LINEAR_RANGE(1220000, 71, 102, 20000), > + REGULATOR_LINEAR_RANGE(1900000, 103, 118, 100000), > +}; In the AXP318W datasheet, it says: section 7.1 DCDC/LCO desgin 8. DCDC6/7/8/9 only able to tune at two voltage ranges which are <1.54v and >1.54v, the tuning voltage should not step cross 1.54v (I translate the original doc into english) so, with this restricition, should we split the range into two? one is dcdc6_lo_range, another dcdc6_hi_range or what do you think? > + > +static const struct regulator_desc axp318_regulators[] = { > + AXP_DESC(AXP318, DCDC1, "dcdc1", "vin19", 1000, 3400, 100, > + AXP318_DCDC1_CONTROL, AXP318_DCDC1_V_OUT_MASK, > + AXP318_DCDC_OUTPUT_CONTROL1, BIT(0)), > + AXP_DESC_RANGES(AXP318, DCDC2, "dcdc2", "vin23", > + axp318_dcdc2_ranges, AXP318_DCDC2_NUM_VOLTAGES, > + AXP318_DCDC2_CONTROL, AXP318_DCDC2_V_OUT_MASK, > + AXP318_DCDC_OUTPUT_CONTROL1, BIT(1)), > + AXP_DESC_RANGES(AXP318, DCDC3, "dcdc3", "vin23", > + axp318_dcdc2_ranges, AXP318_DCDC2_NUM_VOLTAGES, > + AXP318_DCDC3_CONTROL, AXP318_DCDC2_V_OUT_MASK, > + AXP318_DCDC_OUTPUT_CONTROL1, BIT(2)), > + AXP_DESC_RANGES(AXP318, DCDC4, "dcdc4", "vin45", > + axp318_dcdc2_ranges, AXP318_DCDC2_NUM_VOLTAGES, > + AXP318_DCDC4_CONTROL, AXP318_DCDC2_V_OUT_MASK, > + AXP318_DCDC_OUTPUT_CONTROL1, BIT(3)), > + AXP_DESC_RANGES(AXP318, DCDC5, "dcdc5", "vin45", > + axp318_dcdc2_ranges, AXP318_DCDC2_NUM_VOLTAGES, > + AXP318_DCDC5_CONTROL, AXP318_DCDC2_V_OUT_MASK, > + AXP318_DCDC_OUTPUT_CONTROL1, BIT(4)), > + AXP_DESC_RANGES(AXP318, DCDC6, "dcdc6", "vin678", > + axp318_dcdc6_ranges, AXP318_DCDC6_NUM_VOLTAGES, > + AXP318_DCDC6_CONTROL, AXP318_DCDC2_V_OUT_MASK, > + AXP318_DCDC_OUTPUT_CONTROL1, BIT(5)), > + AXP_DESC_RANGES(AXP318, DCDC7, "dcdc7", "vin678", > + axp318_dcdc7_ranges, AXP318_DCDC7_NUM_VOLTAGES, > + AXP318_DCDC7_CONTROL, AXP318_DCDC2_V_OUT_MASK, > + AXP318_DCDC_OUTPUT_CONTROL1, BIT(6)), > + AXP_DESC_RANGES(AXP318, DCDC8, "dcdc8", "vin678", > + axp318_dcdc8_ranges, AXP318_DCDC8_NUM_VOLTAGES, > + AXP318_DCDC8_CONTROL, AXP318_DCDC2_V_OUT_MASK, > + AXP318_DCDC_OUTPUT_CONTROL1, BIT(7)), > + AXP_DESC_RANGES(AXP318, DCDC9, "dcdc9", "vin19", > + axp318_dcdc8_ranges, AXP318_DCDC8_NUM_VOLTAGES, > + AXP318_DCDC9_CONTROL, AXP318_DCDC2_V_OUT_MASK, > + AXP318_DCDC_OUTPUT_CONTROL2, BIT(0)), > + AXP_DESC_SW(AXP318, SWOUT1, "swout1", NULL, > + AXP318_DCDC_OUTPUT_CONTROL2, BIT(3)), > + AXP_DESC_SW(AXP318, SWOUT2, "swout2", NULL, > + AXP318_DCDC_OUTPUT_CONTROL2, BIT(4)), > + AXP_DESC(AXP318, ALDO1, "aldo1", "aldo156in", 500, 3400, 100, > + AXP318_ALDO1_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL1, BIT(0)), > + AXP_DESC(AXP318, ALDO2, "aldo2", "aldo234in", 500, 3400, 100, > + AXP318_ALDO2_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL1, BIT(1)), > + AXP_DESC(AXP318, ALDO3, "aldo3", "aldo234in", 500, 3400, 100, > + AXP318_ALDO3_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL1, BIT(2)), > + AXP_DESC(AXP318, ALDO4, "aldo4", "aldo234in", 500, 3400, 100, > + AXP318_ALDO4_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL1, BIT(3)), > + AXP_DESC(AXP318, ALDO5, "aldo5", "aldo156in", 500, 3400, 100, > + AXP318_ALDO5_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL1, BIT(4)), > + AXP_DESC(AXP318, ALDO6, "aldo6", "aldo156in", 500, 3400, 100, > + AXP318_ALDO6_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL1, BIT(5)), > + AXP_DESC(AXP318, BLDO1, "bldo1", "bldoin", 500, 3400, 100, > + AXP318_BLDO1_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL1, BIT(6)), > + AXP_DESC(AXP318, BLDO2, "bldo2", "bldoin", 500, 3400, 100, > + AXP318_BLDO2_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL1, BIT(7)), > + AXP_DESC(AXP318, BLDO3, "bldo3", "bldoin", 500, 3400, 100, > + AXP318_BLDO3_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL2, BIT(0)), > + AXP_DESC(AXP318, BLDO4, "bldo4", "bldoin", 500, 3400, 100, > + AXP318_BLDO4_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL2, BIT(1)), > + AXP_DESC(AXP318, BLDO5, "bldo5", "bldoin", 500, 3400, 100, > + AXP318_BLDO5_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL2, BIT(2)), > + AXP_DESC(AXP318, CLDO1, "cldo1", "cldoin", 500, 3400, 100, > + AXP318_CLDO1_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL2, BIT(3)), > + AXP_DESC(AXP318, CLDO2, "cldo2", "cldoin", 500, 3400, 100, > + AXP318_CLDO2_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL2, BIT(4)), > + AXP_DESC(AXP318, CLDO3, "cldo3", "cldoin", 500, 3400, 100, > + AXP318_CLDO3_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL2, BIT(5)), > + AXP_DESC(AXP318, CLDO4, "cldo4", "cldoin", 500, 3400, 100, > + AXP318_CLDO4_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL2, BIT(6)), > + AXP_DESC(AXP318, CLDO5, "cldo5", "cldoin", 500, 3400, 100, > + AXP318_CLDO5_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL2, BIT(7)), > + AXP_DESC(AXP318, DLDO1, "dldo1", "dldoin", 500, 3400, 100, > + AXP318_DLDO1_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL3, BIT(0)), > + AXP_DESC(AXP318, DLDO2, "dldo2", "dldoin", 500, 3400, 100, > + AXP318_DLDO2_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL3, BIT(1)), > + AXP_DESC(AXP318, DLDO3, "dldo3", "dldoin", 500, 3400, 100, > + AXP318_DLDO3_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL3, BIT(2)), > + AXP_DESC(AXP318, DLDO4, "dldo4", "dldoin", 500, 3400, 100, > + AXP318_DLDO4_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL3, BIT(3)), > + AXP_DESC(AXP318, DLDO5, "dldo5", "dldoin", 500, 3400, 100, > + AXP318_DLDO5_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL3, BIT(4)), > + AXP_DESC(AXP318, DLDO6, "dldo6", "dldoin", 500, 3400, 100, > + AXP318_DLDO6_CONTROL, AXP318_LDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL3, BIT(5)), .. > + AXP_DESC(AXP318, ELDO1, "eldo1", "eldoin", 500, 1500, 25, > + AXP318_ELDO1_CONTROL, AXP318_ELDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL3, BIT(6)), > + AXP_DESC(AXP318, ELDO2, "eldo2", "eldoin", 500, 1500, 25, > + AXP318_ELDO2_CONTROL, AXP318_ELDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL3, BIT(7)), > + AXP_DESC(AXP318, ELDO3, "eldo3", "eldoin", 500, 1500, 25, > + AXP318_ELDO3_CONTROL, AXP318_ELDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL4, BIT(0)), > + AXP_DESC(AXP318, ELDO4, "eldo4", "eldoin", 500, 1500, 25, > + AXP318_ELDO4_CONTROL, AXP318_ELDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL4, BIT(1)), > + AXP_DESC(AXP318, ELDO5, "eldo5", "eldoin", 500, 1500, 25, > + AXP318_ELDO5_CONTROL, AXP318_ELDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL4, BIT(2)), > + AXP_DESC(AXP318, ELDO6, "eldo6", "eldoin", 500, 1500, 25, > + AXP318_ELDO6_CONTROL, AXP318_ELDO_V_OUT_MASK, > + AXP318_LDO_OUTPUT_CONTROL4, BIT(3)), also, in section 7.1 DCDC/LCO desgin 3. ELDOIN can use DCDC's output as the voltage input, once in this case, the LDO (output?) config voltage should lower than DCDC input voltage. Note: ELDOIN can use PS(Power Supply, should be equal to DCIN) or DCDC as input in case of Radxa A7A (A733) board, it use DCDC9 as ELDOIN, Should we do something in the driver level? or leave up to user > +}; > + > static const struct linear_range axp717_dcdc1_ranges[] = { > REGULATOR_LINEAR_RANGE(500000, 0, 70, 10000), > REGULATOR_LINEAR_RANGE(1220000, 71, 87, 20000), > @@ -1347,6 +1505,7 @@ static int axp20x_set_dcdc_freq(struct platform_device *pdev, u32 dcdcfreq) > step = 150; > break; > case AXP313A_ID: > + case AXP318_ID: > case AXP323_ID: > case AXP717_ID: > case AXP15060_ID: > @@ -1585,6 +1744,10 @@ static int axp20x_regulator_probe(struct platform_device *pdev) > regulators = axp313a_regulators; > nregulators = AXP313A_REG_ID_MAX; > break; > + case AXP318_ID: > + regulators = axp318_regulators; > + nregulators = AXP318_REG_ID_MAX; > + break; > case AXP717_ID: > regulators = axp717_regulators; > nregulators = AXP717_REG_ID_MAX; > @@ -1651,7 +1814,9 @@ static int axp20x_regulator_probe(struct platform_device *pdev) > if ((regulators == axp22x_regulators && i == AXP22X_DC1SW) || > (regulators == axp803_regulators && i == AXP803_DC1SW) || > (regulators == axp809_regulators && i == AXP809_DC1SW) || > - (regulators == axp15060_regulators && i == AXP15060_SW)) { > + (regulators == axp15060_regulators && i == AXP15060_SW) || > + (regulators == axp318_regulators && i == AXP318_SWOUT1) || > + (regulators == axp318_regulators && i == AXP318_SWOUT2)) { > new_desc = devm_kzalloc(&pdev->dev, sizeof(*desc), > GFP_KERNEL); > if (!new_desc) > @@ -1709,7 +1874,8 @@ static int axp20x_regulator_probe(struct platform_device *pdev) > */ > if ((regulators == axp22x_regulators && i == AXP22X_DCDC1) || > (regulators == axp809_regulators && i == AXP809_DCDC1) || > - (regulators == axp15060_regulators && i == AXP15060_DCDC1)) > + (regulators == axp15060_regulators && i == AXP15060_DCDC1) || > + (regulators == axp318_regulators && i == AXP318_DCDC1)) > of_property_read_string(rdev->dev.of_node, > "regulator-name", > &dcdc1_name); > diff --git a/include/linux/mfd/axp20x.h b/include/linux/mfd/axp20x.h > index a871789f6cfa9..9957185458d63 100644 > --- a/include/linux/mfd/axp20x.h > +++ b/include/linux/mfd/axp20x.h > @@ -559,6 +559,49 @@ enum { > AXP313A_REG_ID_MAX, > }; > > +enum { > + AXP318_DCDC1 = 0, > + AXP318_DCDC2, > + AXP318_DCDC3, > + AXP318_DCDC4, > + AXP318_DCDC5, > + AXP318_DCDC6, > + AXP318_DCDC7, > + AXP318_DCDC8, > + AXP318_DCDC9, > + AXP318_ALDO1, > + AXP318_ALDO2, > + AXP318_ALDO3, > + AXP318_ALDO4, > + AXP318_ALDO5, > + AXP318_ALDO6, > + AXP318_BLDO1, > + AXP318_BLDO2, > + AXP318_BLDO3, > + AXP318_BLDO4, > + AXP318_BLDO5, > + AXP318_CLDO1, > + AXP318_CLDO2, > + AXP318_CLDO3, > + AXP318_CLDO4, > + AXP318_CLDO5, > + AXP318_DLDO1, > + AXP318_DLDO2, > + AXP318_DLDO3, > + AXP318_DLDO4, > + AXP318_DLDO5, > + AXP318_DLDO6, > + AXP318_ELDO1, > + AXP318_ELDO2, > + AXP318_ELDO3, > + AXP318_ELDO4, > + AXP318_ELDO5, > + AXP318_ELDO6, > + AXP318_SWOUT1, > + AXP318_SWOUT2, > + AXP318_REG_ID_MAX, > +}; > + > enum { > AXP717_DCDC1 = 0, > AXP717_DCDC2, > -- > 2.25.1 > -- Yixun Lan (dlan)