Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1753832AbdIDOVq (ORCPT ); Mon, 4 Sep 2017 10:21:46 -0400 Received: from mx1.redhat.com ([209.132.183.28]:33956 "EHLO mx1.redhat.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1753725AbdIDOVn (ORCPT ); Mon, 4 Sep 2017 10:21:43 -0400 DMARC-Filter: OpenDMARC Filter v1.3.2 mx1.redhat.com AD1FB75ED3 Authentication-Results: ext-mx04.extmail.prod.ext.phx2.redhat.com; dmarc=none (p=none dis=none) header.from=redhat.com Authentication-Results: ext-mx04.extmail.prod.ext.phx2.redhat.com; spf=fail smtp.mailfrom=hdegoede@redhat.com Subject: Re: [PATCH 02/11] mux: core: Add support for getting a mux controller on a non DT platform To: sathya , MyungJoo Ham , Chanwoo Choi , Guenter Roeck , Heikki Krogerus , Darren Hart , Andy Shevchenko , Peter Rosin , Mathias Nyman Cc: platform-driver-x86@vger.kernel.org, devel@driverdev.osuosl.org, Kuppuswamy Sathyanarayanan , linux-kernel@vger.kernel.org, Greg Kroah-Hartman , linux-usb@vger.kernel.org References: <20170901214845.7153-1-hdegoede@redhat.com> <20170901214845.7153-3-hdegoede@redhat.com> <11e60e4e-31c6-a979-1edd-fd48d4b4cb42@gmail.com> From: Hans de Goede Message-ID: <24913f99-c34f-8bb6-9f30-e1ca06277dd2@redhat.com> Date: Mon, 4 Sep 2017 16:21:39 +0200 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:52.0) Gecko/20100101 Thunderbird/52.2.1 MIME-Version: 1.0 In-Reply-To: <11e60e4e-31c6-a979-1edd-fd48d4b4cb42@gmail.com> Content-Type: text/plain; charset=utf-8; format=flowed Content-Language: en-US Content-Transfer-Encoding: 7bit X-Greylist: Sender IP whitelisted, not delayed by milter-greylist-4.5.16 (mx1.redhat.com [10.5.110.28]); Mon, 04 Sep 2017 14:21:43 +0000 (UTC) Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Content-Length: 6020 Lines: 182 Hi, On 02-09-17 21:13, sathya wrote: > Hi, > > > On 09/01/2017 02:48 PM, Hans de Goede wrote: >> On non DT platforms we cannot get the mux_chip by pnode. Other subsystems >> (regulator, clock, pwm) have the same problem and solve this by allowing >> platform / board-setup code to add entries to a lookup table and then use >> this table to look things up. >> >> This commit adds support for getting a mux controller on a non DT platform >> following this pattern. It is based on a simplified version of the pwm >> subsys lookup code, the dev_id and mux_name parts of a lookup table entry >> are mandatory in the mux-core implementation. >> >> Signed-off-by: Hans de Goede >> --- >> drivers/mux/core.c | 96 +++++++++++++++++++++++++++++++++++++++++++- >> include/linux/mux/consumer.h | 11 +++++ >> 2 files changed, 106 insertions(+), 1 deletion(-) >> >> diff --git a/drivers/mux/core.c b/drivers/mux/core.c >> index 6142493c327b..8864cc745506 100644 >> --- a/drivers/mux/core.c >> +++ b/drivers/mux/core.c >> @@ -24,6 +24,9 @@ >> #include >> #include >> +static DEFINE_MUTEX(mux_lookup_lock); >> +static LIST_HEAD(mux_lookup_list); >> + >> /* >> * The idle-as-is "state" is not an actual state that may be selected, it >> * only implies that the state should not be changed. So, use that state >> @@ -408,6 +411,23 @@ int mux_control_deselect(struct mux_control *mux) >> } >> EXPORT_SYMBOL_GPL(mux_control_deselect); >> +static int parent_name_match(struct device *dev, const void *data) >> +{ >> + const char *parent_name = dev_name(dev->parent); > Device name usually contains id section (devname.id). Did you take this into consideration ? >> + const char *name = data; >> + >> + return strcmp(parent_name, name) == 0; >> +} >> + >> +static struct mux_chip *mux_chip_get_by_name(const char *name) >> +{ >> + struct device *dev; >> + >> + dev = class_find_device(&mux_class, NULL, name, parent_name_match); >> + >> + return dev ? to_mux_chip(dev) : NULL; >> +} >> + >> static int of_dev_node_match(struct device *dev, const void *data) >> { >> return dev->of_node == data; >> @@ -479,6 +499,42 @@ static struct mux_control *of_mux_control_get(struct device *dev, >> } >> /** >> + * mux_add_table() - register PWM device consumers > PWM -> MUX Thank you, good catch, fixed for v2. Regards, Hans >> + * @table: array of consumers to register >> + * @num: number of consumers in table >> + */ >> +void mux_add_table(struct mux_lookup *table, size_t num) >> +{ >> + mutex_lock(&mux_lookup_lock); >> + >> + while (num--) { >> + list_add_tail(&table->list, &mux_lookup_list); >> + table++; >> + } >> + >> + mutex_unlock(&mux_lookup_lock); >> +} >> +EXPORT_SYMBOL_GPL(mux_add_table); >> + >> +/** >> + * mux_remove_table() - unregister PWM device consumers >> + * @table: array of consumers to unregister >> + * @num: number of consumers in table >> + */ >> +void mux_remove_table(struct mux_lookup *table, size_t num) >> +{ >> + mutex_lock(&mux_lookup_lock); >> + >> + while (num--) { >> + list_del(&table->list); >> + table++; >> + } >> + >> + mutex_unlock(&mux_lookup_lock); >> +} >> +EXPORT_SYMBOL_GPL(mux_remove_table); >> + >> +/** >> * mux_control_get() - Get the mux-control for a device. >> * @dev: The device that needs a mux-control. >> * @mux_name: The name identifying the mux-control. >> @@ -487,11 +543,49 @@ static struct mux_control *of_mux_control_get(struct device *dev, >> */ >> struct mux_control *mux_control_get(struct device *dev, const char *mux_name) >> { >> + struct mux_lookup *m, *chosen = NULL; >> + const char *dev_id = dev_name(dev); >> + struct mux_chip *mux_chip; >> + >> /* look up via DT first */ >> if (IS_ENABLED(CONFIG_OF) && dev->of_node) >> return of_mux_control_get(dev, mux_name); >> - return ERR_PTR(-ENODEV); >> + /* >> + * For non DT we look up the provider in the static table typically >> + * provided by board setup code. >> + * >> + * If a match is found, the provider mux chip is looked up by name >> + * and a mux-control is requested using the table provided index. >> + */ >> + mutex_lock(&mux_lookup_lock); >> + list_for_each_entry(m, &mux_lookup_list, list) { >> + if (WARN_ON(!m->dev_id || !m->mux_name || !m->provider)) >> + continue; >> + >> + if (strcmp(m->dev_id, dev_id) == 0 && >> + strcmp(m->mux_name, mux_name) == 0) { >> + chosen = m; >> + break; >> + } >> + } >> + mutex_unlock(&mux_lookup_lock); >> + >> + if (!chosen) >> + return ERR_PTR(-ENODEV); >> + >> + mux_chip = mux_chip_get_by_name(chosen->provider); >> + if (!mux_chip) >> + return ERR_PTR(-EPROBE_DEFER); >> + >> + if (chosen->index >= mux_chip->controllers) { >> + dev_err(dev, "Mux lookup table index out of bounds %u >= %u\n", >> + chosen->index, mux_chip->controllers); >> + put_device(&mux_chip->dev); >> + return ERR_PTR(-EINVAL); >> + } >> + >> + return &mux_chip->mux[chosen->index]; >> } >> EXPORT_SYMBOL_GPL(mux_control_get); >> diff --git a/include/linux/mux/consumer.h b/include/linux/mux/consumer.h >> index ea96d4c82be7..912dd48a3a5d 100644 >> --- a/include/linux/mux/consumer.h >> +++ b/include/linux/mux/consumer.h >> @@ -18,6 +18,17 @@ >> struct device; >> struct mux_control; >> +struct mux_lookup { >> + struct list_head list; >> + const char *provider; >> + unsigned int index; >> + const char *dev_id; >> + const char *mux_name; >> +}; >> + >> +void mux_add_table(struct mux_lookup *table, size_t num); >> +void mux_remove_table(struct mux_lookup *table, size_t num); >> + >> unsigned int mux_control_states(struct mux_control *mux); >> int __must_check mux_control_select(struct mux_control *mux, >> unsigned int state); >