Blame view

arch/arm/plat-omap/mux.c 5.17 KB
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
1
2
3
4
5
  /*
   * linux/arch/arm/plat-omap/mux.c
   *
   * Utility to set the Omap MUX and PULL_DWN registers from a table in mux.h
   *
1a8bfa1eb   Tony Lindgren   [ARM] 3142/1: OMA...
6
   * Copyright (C) 2003 - 2005 Nokia Corporation
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
   *
   * Written by Tony Lindgren <tony.lindgren@nokia.com>
   *
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License as published by
   * the Free Software Foundation; either version 2 of the License, or
   * (at your option) any later version.
   *
   * This program is distributed in the hope that it will be useful,
   * but WITHOUT ANY WARRANTY; without even the implied warranty of
   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
   * GNU General Public License for more details.
   *
   * You should have received a copy of the GNU General Public License
   * along with this program; if not, write to the Free Software
   * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
   *
   */
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
25
26
  #include <linux/module.h>
  #include <linux/init.h>
1a8bfa1eb   Tony Lindgren   [ARM] 3142/1: OMA...
27
  #include <linux/kernel.h>
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
28
29
30
  #include <asm/system.h>
  #include <asm/io.h>
  #include <linux/spinlock.h>
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
31
32
33
  #include <asm/arch/mux.h>
  
  #ifdef CONFIG_OMAP_MUX
1a8bfa1eb   Tony Lindgren   [ARM] 3142/1: OMA...
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
  #define OMAP24XX_L4_BASE	0x48000000
  #define OMAP24XX_PULL_ENA	(1 << 3)
  #define OMAP24XX_PULL_UP	(1 << 4)
  
  static struct pin_config * pin_table;
  static unsigned long pin_table_sz;
  
  extern struct pin_config * omap730_pins;
  extern struct pin_config * omap1xxx_pins;
  extern struct pin_config * omap24xx_pins;
  
  int __init omap_mux_register(struct pin_config * pins, unsigned long size)
  {
  	pin_table = pins;
  	pin_table_sz = size;
  
  	return 0;
  }
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
52
53
54
  /*
   * Sets the Omap MUX and PULL_DWN registers based on the table
   */
1a8bfa1eb   Tony Lindgren   [ARM] 3142/1: OMA...
55
  int __init_or_module omap_cfg_reg(const unsigned long index)
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
56
57
58
59
  {
  	static DEFINE_SPINLOCK(mux_spin_lock);
  
  	unsigned long flags;
1a8bfa1eb   Tony Lindgren   [ARM] 3142/1: OMA...
60
  	struct pin_config *cfg;
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
61
62
63
  	unsigned int reg_orig = 0, reg = 0, pu_pd_orig = 0, pu_pd = 0,
  		pull_orig = 0, pull = 0;
  	unsigned int mask, warn = 0;
1a8bfa1eb   Tony Lindgren   [ARM] 3142/1: OMA...
64
65
  	if (!pin_table)
  		BUG();
92105bb70   Tony Lindgren   [ARM] 2887/1: OMA...
66

1a8bfa1eb   Tony Lindgren   [ARM] 3142/1: OMA...
67
68
69
70
71
72
  	if (index >= pin_table_sz) {
  		printk(KERN_ERR "Invalid pin mux index: %lu (%lu)
  ",
  		       index, pin_table_sz);
  		dump_stack();
  		return -ENODEV;
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
73
  	}
1a8bfa1eb   Tony Lindgren   [ARM] 3142/1: OMA...
74
75
76
77
78
79
80
81
82
  	cfg = (struct pin_config *)&pin_table[index];
  	if (cpu_is_omap24xx()) {
  		u8 reg = 0;
  
  		reg |= cfg->mask & 0x7;
  		if (cfg->pull_val)
  			reg |= OMAP24XX_PULL_ENA;
  		if(cfg->pu_pd_val)
  			reg |= OMAP24XX_PULL_UP;
7bbb3cc5c   Kyungmin Park   ARM: OMAP: 24xx p...
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
  #if defined(CONFIG_OMAP_MUX_DEBUG) || defined(CONFIG_OMAP_MUX_WARNINGS)
  		{
  			u8 orig = omap_readb(OMAP24XX_L4_BASE + cfg->mux_reg);
  			u8 debug = 0;
  
  #ifdef	CONFIG_OMAP_MUX_DEBUG
  			debug = cfg->debug;
  #endif
  			warn = (orig != reg);
  			if (debug || warn)
  				printk("MUX: setup %s (0x%08x): 0x%02x -> 0x%02x
  ",
  						cfg->name,
  						OMAP24XX_L4_BASE + cfg->mux_reg,
  						orig, reg);
  		}
1a8bfa1eb   Tony Lindgren   [ARM] 3142/1: OMA...
99
100
101
102
103
  #endif
  		omap_writeb(reg, OMAP24XX_L4_BASE + cfg->mux_reg);
  
  		return 0;
  	}
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
104
105
106
107
  
  	/* Check the mux register in question */
  	if (cfg->mux_reg) {
  		unsigned	tmp1, tmp2;
bb13b5fdb   Tony Lindgren   [PATCH] ARM: 2804...
108
  		spin_lock_irqsave(&mux_spin_lock, flags);
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
109
110
111
112
113
114
115
116
117
118
119
120
121
122
  		reg_orig = omap_readl(cfg->mux_reg);
  
  		/* The mux registers always seem to be 3 bits long */
  		mask = (0x7 << cfg->mask_offset);
  		tmp1 = reg_orig & mask;
  		reg = reg_orig & ~mask;
  
  		tmp2 = (cfg->mask << cfg->mask_offset);
  		reg |= tmp2;
  
  		if (tmp1 != tmp2)
  			warn = 1;
  
  		omap_writel(reg, cfg->mux_reg);
bb13b5fdb   Tony Lindgren   [PATCH] ARM: 2804...
123
  		spin_unlock_irqrestore(&mux_spin_lock, flags);
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
124
125
126
  	}
  
  	/* Check for pull up or pull down selection on 1610 */
99c658a6c   Vladimir Ananiev   ARM: OMAP: omap15...
127
  	if (!cpu_is_omap15xx()) {
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
128
  		if (cfg->pu_pd_reg && cfg->pull_val) {
bb13b5fdb   Tony Lindgren   [PATCH] ARM: 2804...
129
  			spin_lock_irqsave(&mux_spin_lock, flags);
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
  			pu_pd_orig = omap_readl(cfg->pu_pd_reg);
  			mask = 1 << cfg->pull_bit;
  
  			if (cfg->pu_pd_val) {
  				if (!(pu_pd_orig & mask))
  					warn = 1;
  				/* Use pull up */
  				pu_pd = pu_pd_orig | mask;
  			} else {
  				if (pu_pd_orig & mask)
  					warn = 1;
  				/* Use pull down */
  				pu_pd = pu_pd_orig & ~mask;
  			}
  			omap_writel(pu_pd, cfg->pu_pd_reg);
bb13b5fdb   Tony Lindgren   [PATCH] ARM: 2804...
145
  			spin_unlock_irqrestore(&mux_spin_lock, flags);
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
146
147
148
149
150
  		}
  	}
  
  	/* Check for an associated pull down register */
  	if (cfg->pull_reg) {
bb13b5fdb   Tony Lindgren   [PATCH] ARM: 2804...
151
  		spin_lock_irqsave(&mux_spin_lock, flags);
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
  		pull_orig = omap_readl(cfg->pull_reg);
  		mask = 1 << cfg->pull_bit;
  
  		if (cfg->pull_val) {
  			if (pull_orig & mask)
  				warn = 1;
  			/* Low bit = pull enabled */
  			pull = pull_orig & ~mask;
  		} else {
  			if (!(pull_orig & mask))
  				warn = 1;
  			/* High bit = pull disabled */
  			pull = pull_orig | mask;
  		}
  
  		omap_writel(pull, cfg->pull_reg);
bb13b5fdb   Tony Lindgren   [PATCH] ARM: 2804...
168
  		spin_unlock_irqrestore(&mux_spin_lock, flags);
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
  	}
  
  	if (warn) {
  #ifdef CONFIG_OMAP_MUX_WARNINGS
  		printk(KERN_WARNING "MUX: initialized %s
  ", cfg->name);
  #endif
  	}
  
  #ifdef CONFIG_OMAP_MUX_DEBUG
  	if (cfg->debug || warn) {
  		printk("MUX: Setting register %s
  ", cfg->name);
  		printk("      %s (0x%08x) = 0x%08x -> 0x%08x
  ",
  		       cfg->mux_reg_name, cfg->mux_reg, reg_orig, reg);
99c658a6c   Vladimir Ananiev   ARM: OMAP: omap15...
185
  		if (!cpu_is_omap15xx()) {
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
186
187
188
189
190
191
192
193
194
195
196
197
198
199
  			if (cfg->pu_pd_reg && cfg->pull_val) {
  				printk("      %s (0x%08x) = 0x%08x -> 0x%08x
  ",
  				       cfg->pu_pd_name, cfg->pu_pd_reg,
  				       pu_pd_orig, pu_pd);
  			}
  		}
  
  		if (cfg->pull_reg)
  			printk("      %s (0x%08x) = 0x%08x -> 0x%08x
  ",
  			       cfg->pull_name, cfg->pull_reg, pull_orig, pull);
  	}
  #endif
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
200
201
202
203
204
205
  #ifdef CONFIG_OMAP_MUX_ERRORS
  	return warn ? -ETXTBSY : 0;
  #else
  	return 0;
  #endif
  }
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
206
  EXPORT_SYMBOL(omap_cfg_reg);
1a8bfa1eb   Tony Lindgren   [ARM] 3142/1: OMA...
207
208
209
  #else
  #define omap_mux_init() do {} while(0)
  #define omap_cfg_reg(x)	do {} while(0)
5e1c5ff47   Tony Lindgren   [PATCH] ARM: 2812...
210
  #endif	/* CONFIG_OMAP_MUX */