diff options
author | Sascha Hauer <s.hauer@pengutronix.de> | 2009-04-01 05:11:48 -0400 |
---|---|---|
committer | Sascha Hauer <s.hauer@pengutronix.de> | 2009-05-07 10:14:56 -0400 |
commit | bca6ef1e53b601ece59250f41b9817eba3afa490 (patch) | |
tree | 00b4d295f1b2738ba869ac5750f5f7debd456e01 | |
parent | 01eb1753fcafd25304ec9392e0d9d2acb649931b (diff) |
MXC: Add iomux support for MX35 SoCs
This iomux is called iomux-v3 in the tree because it is the third known
incarnation of MXC iomuxers. It is not only found on the MX35 but also
on the MX51 and probably others.
Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de>
-rw-r--r-- | arch/arm/mach-mx3/Kconfig | 1 | ||||
-rw-r--r-- | arch/arm/plat-mxc/Kconfig | 2 | ||||
-rw-r--r-- | arch/arm/plat-mxc/Makefile | 1 | ||||
-rw-r--r-- | arch/arm/plat-mxc/include/mach/iomux-v3.h | 121 | ||||
-rw-r--r-- | arch/arm/plat-mxc/iomux-v3.c | 98 |
5 files changed, 223 insertions, 0 deletions
diff --git a/arch/arm/mach-mx3/Kconfig b/arch/arm/mach-mx3/Kconfig index 194b8428bba4..6d61ef01db57 100644 --- a/arch/arm/mach-mx3/Kconfig +++ b/arch/arm/mach-mx3/Kconfig | |||
@@ -5,6 +5,7 @@ config ARCH_MX31 | |||
5 | 5 | ||
6 | config ARCH_MX35 | 6 | config ARCH_MX35 |
7 | bool | 7 | bool |
8 | select ARCH_MXC_IOMUX_V3 | ||
8 | 9 | ||
9 | comment "MX3 platforms:" | 10 | comment "MX3 platforms:" |
10 | 11 | ||
diff --git a/arch/arm/plat-mxc/Kconfig b/arch/arm/plat-mxc/Kconfig index 17d0e9906d5f..61acd4ca9349 100644 --- a/arch/arm/plat-mxc/Kconfig +++ b/arch/arm/plat-mxc/Kconfig | |||
@@ -51,4 +51,6 @@ config MXC_PWM | |||
51 | help | 51 | help |
52 | Enable support for the i.MX PWM controller(s). | 52 | Enable support for the i.MX PWM controller(s). |
53 | 53 | ||
54 | config ARCH_MXC_IOMUX_V3 | ||
55 | bool | ||
54 | endif | 56 | endif |
diff --git a/arch/arm/plat-mxc/Makefile b/arch/arm/plat-mxc/Makefile index 055406312b69..e3212c8ff421 100644 --- a/arch/arm/plat-mxc/Makefile +++ b/arch/arm/plat-mxc/Makefile | |||
@@ -7,4 +7,5 @@ obj-y := irq.o clock.o gpio.o time.o devices.o cpu.o system.o | |||
7 | 7 | ||
8 | obj-$(CONFIG_ARCH_MX1) += iomux-mx1-mx2.o dma-mx1-mx2.o | 8 | obj-$(CONFIG_ARCH_MX1) += iomux-mx1-mx2.o dma-mx1-mx2.o |
9 | obj-$(CONFIG_ARCH_MX2) += iomux-mx1-mx2.o dma-mx1-mx2.o | 9 | obj-$(CONFIG_ARCH_MX2) += iomux-mx1-mx2.o dma-mx1-mx2.o |
10 | obj-$(CONFIG_ARCH_MXC_IOMUX_V3) += iomux-v3.o | ||
10 | obj-$(CONFIG_MXC_PWM) += pwm.o | 11 | obj-$(CONFIG_MXC_PWM) += pwm.o |
diff --git a/arch/arm/plat-mxc/include/mach/iomux-v3.h b/arch/arm/plat-mxc/include/mach/iomux-v3.h new file mode 100644 index 000000000000..7cd84547658f --- /dev/null +++ b/arch/arm/plat-mxc/include/mach/iomux-v3.h | |||
@@ -0,0 +1,121 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2009 by Jan Weitzel Phytec Messtechnik GmbH, | ||
3 | * <armlinux@phytec.de> | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or | ||
6 | * modify it under the terms of the GNU General Public License | ||
7 | * as published by the Free Software Foundation; either version 2 | ||
8 | * of the License, or (at your option) any later version. | ||
9 | * This program is distributed in the hope that it will be useful, | ||
10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
12 | * GNU General Public License for more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License | ||
15 | * along with this program; if not, write to the Free Software | ||
16 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, | ||
17 | * MA 02110-1301, USA. | ||
18 | */ | ||
19 | |||
20 | #ifndef __MACH_IOMUX_V3_H__ | ||
21 | #define __MACH_IOMUX_V3_H__ | ||
22 | |||
23 | /* | ||
24 | * build IOMUX_PAD structure | ||
25 | * | ||
26 | * This iomux scheme is based around pads, which are the physical balls | ||
27 | * on the processor. | ||
28 | * | ||
29 | * - Each pad has a pad control register (IOMUXC_SW_PAD_CTRL_x) which controls | ||
30 | * things like driving strength and pullup/pulldown. | ||
31 | * - Each pad can have but not necessarily does have an output routing register | ||
32 | * (IOMUXC_SW_MUX_CTL_PAD_x). | ||
33 | * - Each pad can have but not necessarily does have an input routing register | ||
34 | * (IOMUXC_x_SELECT_INPUT) | ||
35 | * | ||
36 | * The three register sets do not have a fixed offset to each other, | ||
37 | * hence we order this table by pad control registers (which all pads | ||
38 | * have) and put the optional i/o routing registers into additional | ||
39 | * fields. | ||
40 | * | ||
41 | * The naming convention for the pad modes is MX35_PAD_<padname>__<padmode> | ||
42 | * If <padname> or <padmode> refers to a GPIO, it is named | ||
43 | * GPIO_<unit>_<num> | ||
44 | * | ||
45 | */ | ||
46 | |||
47 | struct pad_desc { | ||
48 | unsigned mux_ctrl_ofs:12; /* IOMUXC_SW_MUX_CTL_PAD offset */ | ||
49 | unsigned mux_mode:8; | ||
50 | unsigned pad_ctrl_ofs:12; /* IOMUXC_SW_PAD_CTRL offset */ | ||
51 | #define NO_PAD_CTRL (1 << 16) | ||
52 | unsigned pad_ctrl:17; | ||
53 | unsigned select_input_ofs:12; /* IOMUXC_SELECT_INPUT offset */ | ||
54 | unsigned select_input:3; | ||
55 | }; | ||
56 | |||
57 | #define IOMUX_PAD(_pad_ctrl_ofs, _mux_ctrl_ofs, _mux_mode, _select_input_ofs, \ | ||
58 | _select_input, _pad_ctrl) \ | ||
59 | { \ | ||
60 | .mux_ctrl_ofs = _mux_ctrl_ofs, \ | ||
61 | .mux_mode = _mux_mode, \ | ||
62 | .pad_ctrl_ofs = _pad_ctrl_ofs, \ | ||
63 | .pad_ctrl = _pad_ctrl, \ | ||
64 | .select_input_ofs = _select_input_ofs, \ | ||
65 | .select_input = _select_input, \ | ||
66 | } | ||
67 | |||
68 | /* | ||
69 | * Use to set PAD control | ||
70 | */ | ||
71 | #define PAD_CTL_DRIVE_VOLTAGE_3_3_V 0 | ||
72 | #define PAD_CTL_DRIVE_VOLTAGE_1_8_V 1 | ||
73 | |||
74 | #define PAD_CTL_NO_HYSTERESIS 0 | ||
75 | #define PAD_CTL_HYSTERESIS 1 | ||
76 | |||
77 | #define PAD_CTL_PULL_DISABLED 0x0 | ||
78 | #define PAD_CTL_PULL_KEEPER 0xa | ||
79 | #define PAD_CTL_PULL_DOWN_100K 0xc | ||
80 | #define PAD_CTL_PULL_UP_47K 0xd | ||
81 | #define PAD_CTL_PULL_UP_100K 0xe | ||
82 | #define PAD_CTL_PULL_UP_22K 0xf | ||
83 | |||
84 | #define PAD_CTL_OUTPUT_CMOS 0 | ||
85 | #define PAD_CTL_OUTPUT_OPEN_DRAIN 1 | ||
86 | |||
87 | #define PAD_CTL_DRIVE_STRENGTH_NORM 0 | ||
88 | #define PAD_CTL_DRIVE_STRENGTH_HIGH 1 | ||
89 | #define PAD_CTL_DRIVE_STRENGTH_MAX 2 | ||
90 | |||
91 | #define PAD_CTL_SLEW_RATE_SLOW 0 | ||
92 | #define PAD_CTL_SLEW_RATE_FAST 1 | ||
93 | |||
94 | /* | ||
95 | * setups a single pad: | ||
96 | * - reserves the pad so that it is not claimed by another driver | ||
97 | * - setups the iomux according to the configuration | ||
98 | */ | ||
99 | int mxc_iomux_v3_setup_pad(struct pad_desc *pad); | ||
100 | |||
101 | /* | ||
102 | * setups mutliple pads | ||
103 | * convenient way to call the above function with tables | ||
104 | */ | ||
105 | int mxc_iomux_v3_setup_multiple_pads(struct pad_desc *pad_list, unsigned count); | ||
106 | |||
107 | /* | ||
108 | * releases a single pad: | ||
109 | * - make it available for a future use by another driver | ||
110 | * - DOES NOT reconfigure the IOMUX in its reset state | ||
111 | */ | ||
112 | void mxc_iomux_v3_release_pad(struct pad_desc *pad); | ||
113 | |||
114 | /* | ||
115 | * releases multiple pads | ||
116 | * convenvient way to call the above function with tables | ||
117 | */ | ||
118 | void mxc_iomux_v3_release_multiple_pads(struct pad_desc *pad_list, int count); | ||
119 | |||
120 | #endif /* __MACH_IOMUX_V3_H__*/ | ||
121 | |||
diff --git a/arch/arm/plat-mxc/iomux-v3.c b/arch/arm/plat-mxc/iomux-v3.c new file mode 100644 index 000000000000..77a078f9513f --- /dev/null +++ b/arch/arm/plat-mxc/iomux-v3.c | |||
@@ -0,0 +1,98 @@ | |||
1 | /* | ||
2 | * Copyright 2004-2006 Freescale Semiconductor, Inc. All Rights Reserved. | ||
3 | * Copyright (C) 2008 by Sascha Hauer <kernel@pengutronix.de> | ||
4 | * Copyright (C) 2009 by Jan Weitzel Phytec Messtechnik GmbH, | ||
5 | * <armlinux@phytec.de> | ||
6 | * | ||
7 | * This program is free software; you can redistribute it and/or | ||
8 | * modify it under the terms of the GNU General Public License | ||
9 | * as published by the Free Software Foundation; either version 2 | ||
10 | * of the License, or (at your option) any later version. | ||
11 | * This program is distributed in the hope that it will be useful, | ||
12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
14 | * GNU General Public License for more details. | ||
15 | * | ||
16 | * You should have received a copy of the GNU General Public License | ||
17 | * along with this program; if not, write to the Free Software | ||
18 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, | ||
19 | * MA 02110-1301, USA. | ||
20 | */ | ||
21 | #include <linux/errno.h> | ||
22 | #include <linux/init.h> | ||
23 | #include <linux/kernel.h> | ||
24 | #include <linux/module.h> | ||
25 | #include <linux/string.h> | ||
26 | #include <linux/gpio.h> | ||
27 | |||
28 | #include <mach/hardware.h> | ||
29 | #include <asm/mach/map.h> | ||
30 | #include <mach/iomux-v3.h> | ||
31 | |||
32 | #define IOMUX_BASE IO_ADDRESS(IOMUXC_BASE_ADDR) | ||
33 | |||
34 | static unsigned long iomux_v3_pad_alloc_map[0x200 / BITS_PER_LONG]; | ||
35 | |||
36 | /* | ||
37 | * setups a single pin: | ||
38 | * - reserves the pin so that it is not claimed by another driver | ||
39 | * - setups the iomux according to the configuration | ||
40 | */ | ||
41 | int mxc_iomux_v3_setup_pad(struct pad_desc *pad) | ||
42 | { | ||
43 | unsigned int pad_ofs = pad->pad_ctrl_ofs; | ||
44 | |||
45 | if (test_and_set_bit(pad_ofs >> 2, iomux_v3_pad_alloc_map)) | ||
46 | return -EBUSY; | ||
47 | if (pad->mux_ctrl_ofs) | ||
48 | __raw_writel(pad->mux_mode, IOMUX_BASE + pad->mux_ctrl_ofs); | ||
49 | |||
50 | if (pad->select_input_ofs) | ||
51 | __raw_writel(pad->select_input, | ||
52 | IOMUX_BASE + pad->select_input_ofs); | ||
53 | |||
54 | if (!(pad->pad_ctrl & NO_PAD_CTRL)) | ||
55 | __raw_writel(pad->pad_ctrl, IOMUX_BASE + pad->pad_ctrl_ofs); | ||
56 | return 0; | ||
57 | } | ||
58 | EXPORT_SYMBOL(mxc_iomux_v3_setup_pad); | ||
59 | |||
60 | int mxc_iomux_v3_setup_multiple_pads(struct pad_desc *pad_list, unsigned count) | ||
61 | { | ||
62 | struct pad_desc *p = pad_list; | ||
63 | int i; | ||
64 | int ret; | ||
65 | |||
66 | for (i = 0; i < count; i++) { | ||
67 | ret = mxc_iomux_v3_setup_pad(p); | ||
68 | if (ret) | ||
69 | goto setup_error; | ||
70 | p++; | ||
71 | } | ||
72 | return 0; | ||
73 | |||
74 | setup_error: | ||
75 | mxc_iomux_v3_release_multiple_pads(pad_list, i); | ||
76 | return ret; | ||
77 | } | ||
78 | EXPORT_SYMBOL(mxc_iomux_v3_setup_multiple_pads); | ||
79 | |||
80 | void mxc_iomux_v3_release_pad(struct pad_desc *pad) | ||
81 | { | ||
82 | unsigned int pad_ofs = pad->pad_ctrl_ofs; | ||
83 | |||
84 | clear_bit(pad_ofs >> 2, iomux_v3_pad_alloc_map); | ||
85 | } | ||
86 | EXPORT_SYMBOL(mxc_iomux_v3_release_pad); | ||
87 | |||
88 | void mxc_iomux_v3_release_multiple_pads(struct pad_desc *pad_list, int count) | ||
89 | { | ||
90 | struct pad_desc *p = pad_list; | ||
91 | int i; | ||
92 | |||
93 | for (i = 0; i < count; i++) { | ||
94 | mxc_iomux_v3_release_pad(p); | ||
95 | p++; | ||
96 | } | ||
97 | } | ||
98 | EXPORT_SYMBOL(mxc_iomux_v3_release_multiple_pads); | ||