[PATCH 1/3] pinctrl: pinctrl-imx: add imx pinctrl core driver

Dong Aisheng dongas86 at gmail.com
Sun Apr 15 13:49:04 EST 2012


Hi Sascha,

Thanks for the review.

On Sat, Apr 14, 2012 at 9:48 PM, Sascha Hauer <s.hauer at pengutronix.de> wrote:
> On Sat, Apr 14, 2012 at 12:18:33AM +0800, Dong Aisheng wrote:
>> From: Dong Aisheng <dong.aisheng at linaro.org>
>>
>> The driver has mux and config support while the gpio is still
>> not supported.
>> For select input setting, the driver will handle it internally
>> and do not need user to take care of it.
>>
>> The pinctrl-imx core driver will parse the dts file and dynamically
>> create the pinmux functions and groups.
>>
>> Each IMX SoC pinctrl driver should register pins with a pin register map
>> including mux register and config register and select input map to core
>> for proper operations.
>>
>> +
>> +iomuxc at 020e0000 {
>> +     compatible = "fsl,imx6q-iomuxc";
>> +     reg = <0x020e0000 0x4000>;
>> +
>> +     /* shared pinctrl settings */
>> +     usdhc4 {
>> +             pinctrl_usdhc4_1: usdhc4grp-1 {
>> +                     fsl,pins =      "MX6Q_PAD_SD4_CMD",
>> +                                     "MX6Q_PAD_SD4_CLK",
>> +                                     "MX6Q_PAD_SD4_DAT0",
>> +                                     "MX6Q_PAD_SD4_DAT1",
>> +                                     "MX6Q_PAD_SD4_DAT2",
>> +                                     "MX6Q_PAD_SD4_DAT3",
>> +                                     "MX6Q_PAD_SD4_DAT4",
>> +                                     "MX6Q_PAD_SD4_DAT5",
>> +                                     "MX6Q_PAD_SD4_DAT6",
>> +                                     "MX6Q_PAD_SD4_DAT7";
>
> Do we really want to have all combinations of all pins in the devicetree
> as strings? This is going to be huge.
>
Correct.
My plan is:
1) not all combinations, just define frequently used ones since
it's hard to define all combinations groups for per based IMX.
(maybe we can try to add a fix-up function which can do minor changes
based on available ones, then user does not need to write a new group
caused by only a small change.
what do you think?)

2)  When dt supports macro, will try convert string to integer macro.
Then it maybe:
fsl,pins = <MX6Q_PAD_SD4_CMD 0
                MX6Q_PAD_SD4_CLK 0
                MX6Q_PAD_SD4_DAT0 1
                .................>,
Why i did not start pin id with a raw integer is because it's less
meaning to user and
not easy to read in dts file.

3) maybe we need a separate imx6q-pinctrl.dtsi when it becomes big
after adding all device support.

>> +                     fsl,mux = <0 0 1 1 1 1 1 1 1 1>;
>
> Still we lose the precious information which pin can be put into which
> mode which means that you have to look at the datasheet each time a pin
> is missing.
To avoid this, i also put the pad function macro in driver, user can
look at the pinctrl SoC driver instead of look at the datasheet.
See:
[PATCH 2/3] pinctrl: pinctrl-imx: add imx6q pinctrl drive
+/* imx6q register maps */
+static struct imx_pin_reg imx6q_pin_regs[] = {
+       IMX_PIN_REG(MX6Q_PAD_SD2_DAT1, 0x0360, 0x004C, 0, 0x0000, 0),
/* MX6Q_PAD_SD2_DAT1__USDHC2_DAT1 */
+       IMX_PIN_REG(MX6Q_PAD_SD2_DAT1, 0x0360, 0x004C, 1, 0x0834, 0),
/* MX6Q_PAD_SD2_DAT1__ECSPI5_SS0 */
.......

>
>> +
>> +static int imx_dt_node_to_map(struct pinctrl_dev *pctldev,
>> +                     struct device_node *np,
>> +                     struct pinctrl_map **map, unsigned *num_maps)
>> +{
>> +     struct imx_pmx *ipmx = pinctrl_dev_get_drvdata(pctldev);
>> +     struct imx_pinctrl_info *info = ipmx->info;
>> +     const struct imx_config_properties *imx_config;
>> +     struct pinctrl_map *new_map;
>> +     struct device_node *parent;
>> +     u32 val, config, map_num;
>> +     void *new_config;
>> +     int i, ret;
>> +
>> +     /* first check if there's a config map */
>> +     config = 0;
>> +     for (i = 0; i < info->nconf_properties; i++) {
>> +             imx_config = &info->conf_properties[i];
>> +             ret = of_property_read_u32(np, imx_config->property, &val);
>> +             if (!ret) {
>> +                     dev_dbg(info->dev, "config property %s: 0x%x\n",
>> +                             imx_config->property, val);
>> +                     if (val > imx_config->mask)
>> +                             dev_warn(info->dev, "The config(%s) value 0x%x exceeds the range",
>> +                                     imx_config->property, val);
>> +                     config |= val << imx_config->off;
>> +             }
>> +     }
>> +
>> +     map_num = config ? 2 : 1;
>> +     new_map = kmalloc(sizeof(struct pinctrl_map) * map_num, GFP_KERNEL);
>> +     if (!new_map)
>> +             return -ENOMEM;
>> +
>> +     /* generate mux map */
>> +     parent = of_get_parent(np);
>> +     if (!parent)
>> +             return -EINVAL;
>> +
>> +     new_map[0].type = PIN_MAP_TYPE_MUX_GROUP;
>> +     new_map[0].data.mux.function = parent->name;
>> +     new_map[0].data.mux.group = np->name;
>> +
>> +     if (config) {
>> +             new_config = kmemdup(&config, sizeof(config), GFP_KERNEL);
>
> Check result?
>
Correct. I missed it.

>> +             /* generate config map */
>> +             new_map[1].type = PIN_MAP_TYPE_CONFIGS_GROUP;
>> +             new_map[1].data.configs.group_or_pin = np->name;
>> +             new_map[1].data.configs.configs = (unsigned long *)new_config;
>
> No need to cast void pointers.
>
Thanks, got it.

>> +             new_map[1].data.configs.num_configs = 1;
>> +     }
>> +
>> +     *map = new_map;
>> +     *num_maps = map_num;
>> +
>> +     dev_dbg(pctldev->dev, "map: num %d function %s group %s config 0x%x\n",
>> +             map_num, new_map->data.mux.function,
>> +             new_map->data.mux.group, config);
>> +
>> +     return 0;
>> +}
>> +
>> +static void imx_dt_free_map(struct pinctrl_dev *pctldev,
>> +                             struct pinctrl_map *map, unsigned num_maps)
>> +{
>> +     int i;
>> +
>> +     for (i = 0; i < num_maps; i++)
>> +             if (map[i].type == PIN_MAP_TYPE_CONFIGS_GROUP)
>> +                     kfree(map[i].data.configs.configs);
>> +
>> +     kfree(map);
>> +}
>> +
>> +static struct pinctrl_ops imx_pctrl_ops = {
>> +     .get_groups_count = imx_get_groups_count,
>> +     .get_group_name = imx_get_group_name,
>> +     .get_group_pins = imx_get_group_pins,
>> +     .pin_dbg_show = imx_pin_dbg_show,
>> +     .dt_node_to_map = imx_dt_node_to_map,
>> +     .dt_free_map = imx_dt_free_map,
>> +
>> +};
>> +
>> +static int imx_pmx_enable(struct pinctrl_dev *pctldev, unsigned selector,
>> +                        unsigned group)
>> +{
>> +     struct imx_pmx *ipmx = pinctrl_dev_get_drvdata(pctldev);
>> +     struct imx_pinctrl_info *info = ipmx->info;
>> +     const struct imx_pin_reg *pin_reg;
>> +     const unsigned *pins, *mux;
>> +     unsigned int num_pins, pin_id;
>> +     int i;
>> +
>> +     /*
>> +      * Configure the mux mode for each pin in the group for a specific
>> +      * function.
>> +      */
>> +     pins = info->groups[group].pins;
>> +     num_pins = info->groups[group].num_pins;
>> +     mux = info->groups[group].mux_mode;
>> +
>> +     WARN_ON(!pins || !num_pins || !mux);
>> +
>> +     dev_dbg(ipmx->dev, "enable function %s group %s\n",
>> +             info->functions[selector].name, info->groups[group].name);
>> +
>> +     for (i = 0; i < num_pins; i++) {
>> +             pin_id = pins[i];
>> +
>> +             pin_reg = imx_find_pin_reg(info, pin_id, 1, mux[i]);
>> +             if (!pin_reg)
>> +                     return -EINVAL;
>> +
>> +             if (!pin_reg->mux_reg) {
>> +                     dev_err(ipmx->dev, "Pin(%s) does not support mux function\n",
>> +                             info->pins[pin_id].name);
>> +                     return -EINVAL;
>> +             }
>> +
>> +             writel(mux[i], ipmx->base + pin_reg->mux_reg);
>> +             dev_dbg(ipmx->dev, "write: offset 0x%x val 0x%x\n",
>> +                     pin_reg->mux_reg, mux[i]);
>> +
>> +             /* some pins also need select input setting, set it if found */
>> +             if (pin_reg->input_reg) {
>> +                     writel(pin_reg->input_val, ipmx->base + pin_reg->input_reg);
>> +                     dev_dbg(ipmx->dev,
>> +                             "==>select_input: offset 0x%x val 0x%x\n",
>> +                             pin_reg->input_reg, pin_reg->input_val);
>> +             }
>> +     }
>> +
>> +     return 0;
>> +}
>> +
>> +static void imx_pmx_disable(struct pinctrl_dev *pctldev, unsigned func_selector,
>> +                         unsigned group_selector)
>> +{
>> +     /* nothing to do here */
>> +}
>> +
>> +static int imx_pmx_get_funcs_count(struct pinctrl_dev *pctldev)
>> +{
>> +     struct imx_pmx *ipmx = pinctrl_dev_get_drvdata(pctldev);
>> +     struct imx_pinctrl_info *info = ipmx->info;
>> +
>> +     return info->nfunctions;
>> +}
>> +
>> +static const char *imx_pmx_get_func_name(struct pinctrl_dev *pctldev,
>> +                                       unsigned selector)
>> +{
>> +     struct imx_pmx *ipmx = pinctrl_dev_get_drvdata(pctldev);
>> +     struct imx_pinctrl_info *info = ipmx->info;
>> +
>> +     return info->functions[selector].name;
>> +}
>> +
>> +static int imx_pmx_get_groups(struct pinctrl_dev *pctldev, unsigned selector,
>> +                            const char * const **groups,
>> +                            unsigned * const num_groups)
>> +{
>> +     struct imx_pmx *ipmx = pinctrl_dev_get_drvdata(pctldev);
>> +     struct imx_pinctrl_info *info = ipmx->info;
>> +
>> +     *groups = info->functions[selector].groups;
>> +     *num_groups = info->functions[selector].num_groups;
>> +
>> +     return 0;
>> +}
>> +
>> +static struct pinmux_ops imx_pmx_ops = {
>> +     .get_functions_count = imx_pmx_get_funcs_count,
>> +     .get_function_name = imx_pmx_get_func_name,
>> +     .get_function_groups = imx_pmx_get_groups,
>> +     .enable = imx_pmx_enable,
>> +     .disable = imx_pmx_disable,
>> +};
>> +
>> +static int imx_pinconf_get(struct pinctrl_dev *pctldev,
>> +                          unsigned pin_id, unsigned long *config)
>> +{
>> +     struct imx_pmx *ipmx = pinctrl_dev_get_drvdata(pctldev);
>> +     struct imx_pinctrl_info *info = ipmx->info;
>> +     const struct imx_pin_reg *pin_reg;
>> +
>> +     pin_reg = imx_find_pin_reg(info, pin_id, 0, 0);
>> +     if (!pin_reg)
>> +             return -EINVAL;
>> +
>> +     if (!pin_reg->conf_reg) {
>> +             dev_err(info->dev, "Pin(%s) does not support config function\n",
>> +                     info->pins[pin_id].name);
>> +             return -EINVAL;
>> +     }
>> +
>> +     *config = readl(ipmx->base + pin_reg->conf_reg);
>> +
>> +     return 0;
>> +}
>> +
>> +static int imx_pinconf_set(struct pinctrl_dev *pctldev,
>> +                          unsigned pin_id, unsigned long config)
>> +{
>> +     struct imx_pmx *ipmx = pinctrl_dev_get_drvdata(pctldev);
>> +     struct imx_pinctrl_info *info = ipmx->info;
>> +     const struct imx_pin_reg *pin_reg;
>> +
>> +     pin_reg = imx_find_pin_reg(info, pin_id, 0, 0);
>> +     if (!pin_reg)
>> +             return -EINVAL;
>> +
>> +     if (!pin_reg->conf_reg) {
>> +             dev_err(info->dev, "Pin(%s) does not support config function\n",
>> +                     info->pins[pin_id].name);
>> +             return -EINVAL;
>> +     }
>> +
>> +     writel(config, ipmx->base + pin_reg->conf_reg);
>> +
>> +     return 0;
>> +}
>> +
>> +static int imx_pinconf_group_get(struct pinctrl_dev *pctldev,
>> +                                unsigned group, unsigned long *config)
>> +{
>> +     struct imx_pmx *ipmx = pinctrl_dev_get_drvdata(pctldev);
>> +     struct imx_pinctrl_info *info = ipmx->info;
>> +
>> +     *config = info->groups[group].config;
>> +     dev_dbg(ipmx->dev, "get group %s config 0x%lx\n",
>> +             info->groups[group].name, *config);
>> +
>> +     return 0;
>> +}
>> +
>> +static int imx_pinconf_group_set(struct pinctrl_dev *pctldev,
>> +                                unsigned group, unsigned long config)
>> +{
>> +     struct imx_pmx *ipmx = pinctrl_dev_get_drvdata(pctldev);
>> +     struct imx_pinctrl_info *info = ipmx->info;
>> +     const struct imx_pin_reg *pin_reg;
>> +     const unsigned *pins;
>> +     unsigned int num_pins, pin_id;
>> +     int i;
>> +
>> +     pins = info->groups[group].pins;
>> +     num_pins = info->groups[group].num_pins;
>> +
>> +     WARN_ON(!pins || !num_pins);
>> +
>> +     dev_dbg(ipmx->dev, "set group %s config 0x%lx\n",
>> +             info->groups[group].name, config);
>> +
>> +     for (i = 0; i < num_pins; i++) {
>> +             pin_id = pins[i];
>> +
>> +             pin_reg = imx_find_pin_reg(info, pin_id, 0, 0);
>> +             if (!pin_reg)
>> +                     return -EINVAL;
>> +
>> +             if (!pin_reg->conf_reg) {
>> +                     dev_err(ipmx->dev, "Pin(%s) does not support config function\n",
>> +                             info->pins[pin_id].name);
>> +                     return -EINVAL;
>> +             }
>> +
>> +             writel(config, ipmx->base + pin_reg->conf_reg);
>> +             dev_dbg(ipmx->dev, "write: offset 0x%x val 0x%lx\n",
>> +                     pin_reg->conf_reg, config);
>> +     }
>> +
>> +     /* save the current group config value */
>> +     info->groups[group].config = config;
>> +
>> +     return 0;
>> +}
>> +
>> +static void imx_pinconf_dbg_show(struct pinctrl_dev *pctldev,
>> +                                struct seq_file *s, unsigned pin_id)
>> +{
>> +     struct imx_pmx *ipmx = pinctrl_dev_get_drvdata(pctldev);
>> +     struct imx_pinctrl_info *info = ipmx->info;
>> +     const struct imx_pin_reg *pin_reg;
>> +     unsigned long config;
>> +
>> +     pin_reg = imx_find_pin_reg(info, pin_id, 0, 0);
>> +     if (!pin_reg) {
>> +             seq_printf(s, "N/A");
>> +             return;
>> +     }
>> +
>> +     config = readl(ipmx->base + pin_reg->conf_reg);
>> +     seq_printf(s, "0x%lx", config);
>> +}
>> +
>> +static void imx_pinconf_group_dbg_show(struct pinctrl_dev *pctldev,
>> +                                      struct seq_file *s, unsigned group)
>> +{
>> +     unsigned long config;
>> +     int ret;
>> +
>> +     ret = imx_pinconf_group_get(pctldev, group, &config);
>> +     if (!ret)
>> +             seq_printf(s, "0x%lx", config);
>> +}
>> +
>> +struct pinconf_ops imx_pinconf_ops = {
>> +     .pin_config_get = imx_pinconf_get,
>> +     .pin_config_set = imx_pinconf_set,
>> +     .pin_config_group_get = imx_pinconf_group_get,
>> +     .pin_config_group_set = imx_pinconf_group_set,
>> +     .pin_config_dbg_show = imx_pinconf_dbg_show,
>> +     .pin_config_group_dbg_show = imx_pinconf_group_dbg_show,
>> +};
>> +
>> +static struct pinctrl_desc imx_pmx_desc = {
>> +     .name = DRIVER_NAME,
>> +     .pctlops = &imx_pctrl_ops,
>> +     .pmxops = &imx_pmx_ops,
>> +     .confops = &imx_pinconf_ops,
>> +     .owner = THIS_MODULE,
>> +};
>> +
>> +#ifdef CONFIG_OF
>> +static unsigned int imx_pmx_pin_name_to_id(struct imx_pinctrl_info *info,
>> +                                     const char *s)
>> +{
>> +     unsigned int i;
>> +     const struct pinctrl_pin_desc *desc;
>> +
>> +     BUG_ON(!info || !info->pins || !info->npins);
>> +
>> +     for (i = 0; i < info->npins; i++) {
>> +             desc = info->pins + i;
>> +             if (desc->name && !strcmp(s, desc->name))
>> +                     return desc->number;
>> +     }
>> +
>> +     return -EINVAL;
>> +}
>> +
>> +static int __devinit imx_pmx_parse_groups(struct device_node *np,
>> +                             struct imx_pin_group *grp,
>> +                             struct imx_pinctrl_info *info, u32 index)
>> +{
>> +     struct property *prop;
>> +     const char *s;
>> +     int ret, len;
>> +     u32 i = 0;
>> +
>> +     dev_dbg(info->dev, "group(%d): %s\n", index, np->name);
>> +
>> +     /* Initialise group */
>> +     grp->name = np->name;
>> +     grp->num_pins = of_property_count_strings(np, "fsl,pins");
>> +     if (grp->num_pins < 0) {
>> +             dev_err(info->dev, "failed to get fsl,pins\n");
>> +             return grp->num_pins;
>> +     }
>> +
>> +     grp->pins = devm_kzalloc(info->dev, grp->num_pins * sizeof(unsigned int),
>> +                             GFP_KERNEL);
>> +     grp->mux_mode = devm_kzalloc(info->dev, grp->num_pins * sizeof(unsigned int),
>> +                             GFP_KERNEL);
>> +
>> +     of_property_for_each_string(np, "fsl,pins", prop, s) {
>> +             ret = imx_pmx_pin_name_to_id(info, s);
>> +             if (ret < 0) {
>> +                     dev_err(info->dev, "failed to get pin: %s\n", s);
>> +                     return ret;
>> +             }
>> +             grp->pins[i++] = ret;
>> +     }
>> +
>> +     if (i != grp->num_pins) {
>> +             dev_err(info->dev, "parsed wrong pins number?\n");
>> +             return -EINVAL;
>> +     }
>> +
>> +     /* group sanity check */
>> +     if (of_get_property(np, "fsl,mux", &len) &&
>> +             len != grp->num_pins * sizeof(unsigned int)) {
>> +             dev_err(info->dev, "fsl,mux length is not equal to fsl,pins\n");
>> +             return -EINVAL;
>> +     }
>> +
>> +     ret = of_property_read_u32_array(np, "fsl,mux",
>> +                     grp->mux_mode, grp->num_pins);
>> +     if (ret) {
>> +             dev_err(info->dev, "failed to get fsl,mux\n");
>> +             return ret;
>> +     }
>> +
>> +#ifdef DEBUG
>> +     IMX_PMX_DUMP("pins", grp->pins, grp->num_pins);
>> +     IMX_PMX_DUMP("mux", grp->mux_mode, grp->num_pins);
>> +#endif
>> +     return 0;
>> +}
>> +
>> +static int __devinit imx_pmx_parse_functions(struct device_node *np,
>> +                     struct imx_pinctrl_info *info, u32 index)
>> +{
>> +     struct device_node *child;
>> +     struct imx_pmx_func *func;
>> +     struct imx_pin_group *grp;
>> +     int ret;
>> +     static u32 grp_index;
>> +     u32 i = 0;
>> +
>> +     dev_dbg(info->dev, "parse function(%d): %s\n", index, np->name);
>> +
>> +     func = &info->functions[index];
>> +
>> +     /* Initialise function */
>> +     func->name = np->name;
>> +     func->num_groups = of_get_child_count(np);
>> +     if (func->num_groups <= 0) {
>> +             dev_err(info->dev, "no groups defined\n");
>> +             return -EINVAL;
>> +     }
>> +     func->groups = devm_kzalloc(info->dev,
>> +                     func->num_groups * sizeof(char *), GFP_KERNEL);
>> +
>> +     for_each_child_of_node(np, child) {
>> +             func->groups[i] = child->name;
>> +             grp = &info->groups[grp_index++];
>> +             ret = imx_pmx_parse_groups(child, grp, info, i++);
>> +             if (ret)
>> +                     return ret;
>> +     }
>> +
>> +     if (i != func->num_groups) {
>> +             dev_err(info->dev, "parsed wrong groups number?\n");
>> +             return -EINVAL;
>> +     }
>> +
>> +     return 0;
>> +}
>> +
>> +static int __devinit imx_pmx_probe_dt(struct platform_device *pdev,
>> +                             struct imx_pinctrl_info *info)
>> +{
>> +     struct device_node *np = pdev->dev.of_node;
>> +     struct device_node *child;
>> +     int ret;
>> +     u32 nfuncs = 0;
>> +     u32 i = 0;
>> +
>> +     if (!np)
>> +             return -ENODEV;
>> +
>> +     nfuncs = of_get_child_count(np);
>> +     if (nfuncs <= 0) {
>> +             dev_err(&pdev->dev, "no functions defined\n");
>> +             return -EINVAL;
>> +     }
>> +
>> +     info->nfunctions = nfuncs;
>> +     info->functions = devm_kzalloc(&pdev->dev, nfuncs * sizeof(struct imx_pmx_func),
>> +                                     GFP_KERNEL);
>> +     if (!info->functions)
>> +             return -ENOMEM;
>> +
>> +     info->ngroups = 0;
>> +     for_each_child_of_node(np, child)
>> +             info->ngroups += of_get_child_count(child);
>> +     info->groups = devm_kzalloc(&pdev->dev, nfuncs * sizeof(struct imx_pin_group),
>> +                                     GFP_KERNEL);
>> +     if (!info->groups)
>> +             return -ENOMEM;
>> +
>> +     for_each_child_of_node(np, child) {
>> +             ret = imx_pmx_parse_functions(child, info, i++);
>> +             if (ret) {
>> +                     dev_err(&pdev->dev, "failed to parse function\n");
>> +                     return ret;
>> +             }
>> +     }
>> +
>> +     if (i != info->nfunctions) {
>> +             dev_err(info->dev, "parsed wrong functions number?\n");
>> +             return -EINVAL;
>> +     }
>> +
>> +     return 0;
>> +}
>> +#else
>> +static int __devinit imx_pmx_probe_dt(struct platform_device *pdev,
>> +                             struct imx_pinctrl_info *info)
>> +{
>> +     return -ENODEV;
>> +}
>> +#endif
>> +
>> +static inline void imx_pmx_desc_init(struct pinctrl_desc *pmx_desc,
>> +                             const struct imx_pinctrl_info *info)
>> +{
>> +     pmx_desc->pins = info->pins;
>> +     pmx_desc->npins = info->npins;
>> +}
>> +
>> +static const struct of_device_id imx_pmx_dt_ids[] = {
>> +     { /* sentinel */ }
>> +};
>> +MODULE_DEVICE_TABLE(of, imx_pmx_dt_ids);
>> +
>> +static int __init imx_pmx_probe(struct platform_device *pdev)
>> +{
>> +     const struct of_device_id *of_id =
>> +                     of_match_device(imx_pmx_dt_ids, &pdev->dev);
>> +     struct device *dev = &pdev->dev;
>> +     struct imx_pmx *ipmx;
>> +     struct resource *res;
>> +     struct imx_pinctrl_info *info;
>> +     resource_size_t res_size;
>> +     int ret;
>> +
>> +     info = of_id->data;
>> +     if (!info || !info->pins || !info->npins
>> +               || !info->pin_regs || !info->npin_regs
>> +               || !info->conf_properties || !info->nconf_properties) {
>> +             dev_err(&pdev->dev, "wrong pinctrl info\n");
>> +             return -EINVAL;
>> +     }
>> +     info->dev = &pdev->dev;
>> +
>> +     /* Create state holders etc for this driver */
>> +     ipmx = devm_kzalloc(&pdev->dev, sizeof(*ipmx), GFP_KERNEL);
>> +     if (!ipmx)
>> +             return -ENOMEM;
>> +
>> +     res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
>> +     if (!res)
>> +             return -ENOENT;
>> +
>> +     res_size = resource_size(res);
>> +     if (!devm_request_mem_region(dev, res->start, res_size, res->name))
>> +             return -EBUSY;
>> +
>> +     ipmx->base = devm_ioremap_nocache(dev, res->start, res_size);
>> +     if (!ipmx->base)
>> +             return -EBUSY;
>> +
>> +     imx_pmx_desc_init(&imx_pmx_desc, info);
>> +     ret = imx_pmx_probe_dt(pdev, info);
>> +     if (ret) {
>> +             dev_err(&pdev->dev, "fail to probe dt properties\n");
>> +             return ret;
>> +     }
>
> So this driver only works with dt? Why can it be compiled without it
> then?
>
Yes,
The driver was first design to support both dt and non-dt (dt first).
Currently it only supports dt.
i can change the dependency in Kconfig  to make sure it can not
compile without dt.

Regards
Dong Aisheng


More information about the devicetree-discuss mailing list