On Sat, Apr 14, 2012 at 12:18:33AM +0800, Dong Aisheng wrote:
> From: Dong Aisheng <[email protected]>
> 
> 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@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.

> +                     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.

> +
> +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?

> +             /* 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.

> +             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?

Sascha

-- 
Pengutronix e.K.                           |                             |
Industrial Linux Solutions                 | http://www.pengutronix.de/  |
Peiner Str. 6-8, 31137 Hildesheim, Germany | Phone: +49-5121-206917-0    |
Amtsgericht Hildesheim, HRA 2686           | Fax:   +49-5121-206917-5555 |
_______________________________________________
devicetree-discuss mailing list
[email protected]
https://lists.ozlabs.org/listinfo/devicetree-discuss

Reply via email to