mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-08-05 16:54:27 +00:00

The following structures are not modified in these drivers. - struct meson_bank - struct meson_pmx_bank - struct meson_pmx_func - struct meson_pmx_group - struct meson_pinctrl_data - struct meson_axg_pmx_data Constifying these structures moves some data to a read-only section, so increase overall security. On a x86_64, with allmodconfig: Before: ====== text data bss dec hex filename 10818 11696 0 22514 57f2 drivers/pinctrl/meson/pinctrl-amlogic-c3.o 17198 17680 0 34878 883e drivers/pinctrl/meson/pinctrl-amlogic-t7.o 14161 11200 0 25361 6311 drivers/pinctrl/meson/pinctrl-meson8b.o 17348 12512 0 29860 74a4 drivers/pinctrl/meson/pinctrl-meson8.o 3070 324 0 3394 d42 drivers/pinctrl/meson/pinctrl-meson8-pmx.o 9317 9648 0 18965 4a15 drivers/pinctrl/meson/pinctrl-meson-a1.o 12115 11664 0 23779 5ce3 drivers/pinctrl/meson/pinctrl-meson-axg.o 2470 120 0 2590 a1e drivers/pinctrl/meson/pinctrl-meson-axg-pmx.o 15125 15224 0 30349 768d drivers/pinctrl/meson/pinctrl-meson-g12a.o 13800 10160 0 23960 5d98 drivers/pinctrl/meson/pinctrl-meson-gxbb.o 13040 9648 0 22688 58a0 drivers/pinctrl/meson/pinctrl-meson-gxl.o 20507 1132 48 21687 54b7 drivers/pinctrl/meson/pinctrl-meson.o 12212 12880 0 25092 6204 drivers/pinctrl/meson/pinctrl-meson-s4.o After: ===== text data bss dec hex filename 22242 248 0 22490 57da drivers/pinctrl/meson/pinctrl-amlogic-c3.o 34638 248 0 34886 8846 drivers/pinctrl/meson/pinctrl-amlogic-t7.o 25137 232 0 25369 6319 drivers/pinctrl/meson/pinctrl-meson8b.o 29604 232 0 29836 748c drivers/pinctrl/meson/pinctrl-meson8.o 3070 324 0 3394 d42 drivers/pinctrl/meson/pinctrl-meson8-pmx.o 18725 248 0 18973 4a1d drivers/pinctrl/meson/pinctrl-meson-a1.o 23539 248 0 23787 5ceb drivers/pinctrl/meson/pinctrl-meson-axg.o 2470 120 0 2590 a1e drivers/pinctrl/meson/pinctrl-meson-axg-pmx.o 30101 256 0 30357 7695 drivers/pinctrl/meson/pinctrl-meson-g12a.o 23688 248 0 23936 5d80 drivers/pinctrl/meson/pinctrl-meson-gxbb.o 22416 248 0 22664 5888 drivers/pinctrl/meson/pinctrl-meson-gxl.o 20507 1132 48 21687 54b7 drivers/pinctrl/meson/pinctrl-meson.o 24820 248 0 25068 61ec drivers/pinctrl/meson/pinctrl-meson-s4.o Signed-off-by: Christophe JAILLET <christophe.jaillet@wanadoo.fr> Reviewed-by: Jerome Brunet <jbrunet@baylibre.com> Tested-by: Jerome Brunet <jbrunet@baylibre.com> Link: https://lore.kernel.org/f74e326bd7d48003c06219545bad7c2ef1a84bf8.1723053850.git.christophe.jaillet@wanadoo.fr Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
105 lines
3.1 KiB
C
105 lines
3.1 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* First generation of pinmux driver for Amlogic Meson SoCs
|
|
*
|
|
* Copyright (C) 2014 Beniamino Galvani <b.galvani@gmail.com>
|
|
* Copyright (C) 2017 Jerome Brunet <jbrunet@baylibre.com>
|
|
*/
|
|
|
|
/* For this first generation of pinctrl driver every pinmux group can be
|
|
* enabled by a specific bit in the first register range. When all groups for
|
|
* a given pin are disabled the pin acts as a GPIO.
|
|
*/
|
|
#include <linux/device.h>
|
|
#include <linux/regmap.h>
|
|
#include <linux/pinctrl/pinctrl.h>
|
|
#include <linux/pinctrl/pinmux.h>
|
|
|
|
#include "pinctrl-meson.h"
|
|
#include "pinctrl-meson8-pmx.h"
|
|
|
|
/**
|
|
* meson8_pmx_disable_other_groups() - disable other groups using a given pin
|
|
*
|
|
* @pc: meson pin controller device
|
|
* @pin: number of the pin
|
|
* @sel_group: index of the selected group, or -1 if none
|
|
*
|
|
* The function disables all pinmux groups using a pin except the
|
|
* selected one. If @sel_group is -1 all groups are disabled, leaving
|
|
* the pin in GPIO mode.
|
|
*/
|
|
static void meson8_pmx_disable_other_groups(struct meson_pinctrl *pc,
|
|
unsigned int pin, int sel_group)
|
|
{
|
|
const struct meson_pmx_group *group;
|
|
struct meson8_pmx_data *pmx_data;
|
|
int i, j;
|
|
|
|
for (i = 0; i < pc->data->num_groups; i++) {
|
|
group = &pc->data->groups[i];
|
|
pmx_data = (struct meson8_pmx_data *)group->data;
|
|
if (pmx_data->is_gpio || i == sel_group)
|
|
continue;
|
|
|
|
for (j = 0; j < group->num_pins; j++) {
|
|
if (group->pins[j] == pin) {
|
|
/* We have found a group using the pin */
|
|
regmap_update_bits(pc->reg_mux,
|
|
pmx_data->reg * 4,
|
|
BIT(pmx_data->bit), 0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static int meson8_pmx_set_mux(struct pinctrl_dev *pcdev, unsigned func_num,
|
|
unsigned group_num)
|
|
{
|
|
struct meson_pinctrl *pc = pinctrl_dev_get_drvdata(pcdev);
|
|
const struct meson_pmx_func *func = &pc->data->funcs[func_num];
|
|
const struct meson_pmx_group *group = &pc->data->groups[group_num];
|
|
struct meson8_pmx_data *pmx_data =
|
|
(struct meson8_pmx_data *)group->data;
|
|
int i, ret = 0;
|
|
|
|
dev_dbg(pc->dev, "enable function %s, group %s\n", func->name,
|
|
group->name);
|
|
|
|
/*
|
|
* Disable groups using the same pin.
|
|
* The selected group is not disabled to avoid glitches.
|
|
*/
|
|
for (i = 0; i < group->num_pins; i++)
|
|
meson8_pmx_disable_other_groups(pc, group->pins[i], group_num);
|
|
|
|
/* Function 0 (GPIO) doesn't need any additional setting */
|
|
if (func_num)
|
|
ret = regmap_update_bits(pc->reg_mux, pmx_data->reg * 4,
|
|
BIT(pmx_data->bit),
|
|
BIT(pmx_data->bit));
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int meson8_pmx_request_gpio(struct pinctrl_dev *pcdev,
|
|
struct pinctrl_gpio_range *range,
|
|
unsigned offset)
|
|
{
|
|
struct meson_pinctrl *pc = pinctrl_dev_get_drvdata(pcdev);
|
|
|
|
meson8_pmx_disable_other_groups(pc, offset, -1);
|
|
|
|
return 0;
|
|
}
|
|
|
|
const struct pinmux_ops meson8_pmx_ops = {
|
|
.set_mux = meson8_pmx_set_mux,
|
|
.get_functions_count = meson_pmx_get_funcs_count,
|
|
.get_function_name = meson_pmx_get_func_name,
|
|
.get_function_groups = meson_pmx_get_groups,
|
|
.gpio_request_enable = meson8_pmx_request_gpio,
|
|
};
|
|
EXPORT_SYMBOL_GPL(meson8_pmx_ops);
|
|
MODULE_DESCRIPTION("Amlogic Meson SoCs first generation pinmux driver");
|
|
MODULE_LICENSE("GPL v2");
|