xref: /OK3568_Linux_fs/kernel/drivers/pcmcia/soc_common.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*======================================================================
2*4882a593Smuzhiyun 
3*4882a593Smuzhiyun     Common support code for the PCMCIA control functionality of
4*4882a593Smuzhiyun     integrated SOCs like the SA-11x0 and PXA2xx microprocessors.
5*4882a593Smuzhiyun 
6*4882a593Smuzhiyun     The contents of this file are subject to the Mozilla Public
7*4882a593Smuzhiyun     License Version 1.1 (the "License"); you may not use this file
8*4882a593Smuzhiyun     except in compliance with the License. You may obtain a copy of
9*4882a593Smuzhiyun     the License at http://www.mozilla.org/MPL/
10*4882a593Smuzhiyun 
11*4882a593Smuzhiyun     Software distributed under the License is distributed on an "AS
12*4882a593Smuzhiyun     IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
13*4882a593Smuzhiyun     implied. See the License for the specific language governing
14*4882a593Smuzhiyun     rights and limitations under the License.
15*4882a593Smuzhiyun 
16*4882a593Smuzhiyun     The initial developer of the original code is John G. Dorsey
17*4882a593Smuzhiyun     <john+@cs.cmu.edu>.  Portions created by John G. Dorsey are
18*4882a593Smuzhiyun     Copyright (C) 1999 John G. Dorsey.  All Rights Reserved.
19*4882a593Smuzhiyun 
20*4882a593Smuzhiyun     Alternatively, the contents of this file may be used under the
21*4882a593Smuzhiyun     terms of the GNU Public License version 2 (the "GPL"), in which
22*4882a593Smuzhiyun     case the provisions of the GPL are applicable instead of the
23*4882a593Smuzhiyun     above.  If you wish to allow the use of your version of this file
24*4882a593Smuzhiyun     only under the terms of the GPL and not to allow others to use
25*4882a593Smuzhiyun     your version of this file under the MPL, indicate your decision
26*4882a593Smuzhiyun     by deleting the provisions above and replace them with the notice
27*4882a593Smuzhiyun     and other provisions required by the GPL.  If you do not delete
28*4882a593Smuzhiyun     the provisions above, a recipient may use your version of this
29*4882a593Smuzhiyun     file under either the MPL or the GPL.
30*4882a593Smuzhiyun 
31*4882a593Smuzhiyun ======================================================================*/
32*4882a593Smuzhiyun 
33*4882a593Smuzhiyun 
34*4882a593Smuzhiyun #include <linux/cpufreq.h>
35*4882a593Smuzhiyun #include <linux/gpio.h>
36*4882a593Smuzhiyun #include <linux/gpio/consumer.h>
37*4882a593Smuzhiyun #include <linux/init.h>
38*4882a593Smuzhiyun #include <linux/interrupt.h>
39*4882a593Smuzhiyun #include <linux/io.h>
40*4882a593Smuzhiyun #include <linux/irq.h>
41*4882a593Smuzhiyun #include <linux/kernel.h>
42*4882a593Smuzhiyun #include <linux/mm.h>
43*4882a593Smuzhiyun #include <linux/module.h>
44*4882a593Smuzhiyun #include <linux/moduleparam.h>
45*4882a593Smuzhiyun #include <linux/mutex.h>
46*4882a593Smuzhiyun #include <linux/regulator/consumer.h>
47*4882a593Smuzhiyun #include <linux/spinlock.h>
48*4882a593Smuzhiyun #include <linux/timer.h>
49*4882a593Smuzhiyun 
50*4882a593Smuzhiyun #include <mach/hardware.h>
51*4882a593Smuzhiyun 
52*4882a593Smuzhiyun #include "soc_common.h"
53*4882a593Smuzhiyun 
54*4882a593Smuzhiyun static irqreturn_t soc_common_pcmcia_interrupt(int irq, void *dev);
55*4882a593Smuzhiyun 
56*4882a593Smuzhiyun #ifdef CONFIG_PCMCIA_DEBUG
57*4882a593Smuzhiyun 
58*4882a593Smuzhiyun static int pc_debug;
59*4882a593Smuzhiyun module_param(pc_debug, int, 0644);
60*4882a593Smuzhiyun 
soc_pcmcia_debug(struct soc_pcmcia_socket * skt,const char * func,int lvl,const char * fmt,...)61*4882a593Smuzhiyun void soc_pcmcia_debug(struct soc_pcmcia_socket *skt, const char *func,
62*4882a593Smuzhiyun 		      int lvl, const char *fmt, ...)
63*4882a593Smuzhiyun {
64*4882a593Smuzhiyun 	struct va_format vaf;
65*4882a593Smuzhiyun 	va_list args;
66*4882a593Smuzhiyun 	if (pc_debug > lvl) {
67*4882a593Smuzhiyun 		va_start(args, fmt);
68*4882a593Smuzhiyun 
69*4882a593Smuzhiyun 		vaf.fmt = fmt;
70*4882a593Smuzhiyun 		vaf.va = &args;
71*4882a593Smuzhiyun 
72*4882a593Smuzhiyun 		printk(KERN_DEBUG "skt%u: %s: %pV", skt->nr, func, &vaf);
73*4882a593Smuzhiyun 
74*4882a593Smuzhiyun 		va_end(args);
75*4882a593Smuzhiyun 	}
76*4882a593Smuzhiyun }
77*4882a593Smuzhiyun EXPORT_SYMBOL(soc_pcmcia_debug);
78*4882a593Smuzhiyun 
79*4882a593Smuzhiyun #endif
80*4882a593Smuzhiyun 
81*4882a593Smuzhiyun #define to_soc_pcmcia_socket(x)	\
82*4882a593Smuzhiyun 	container_of(x, struct soc_pcmcia_socket, socket)
83*4882a593Smuzhiyun 
soc_pcmcia_regulator_set(struct soc_pcmcia_socket * skt,struct soc_pcmcia_regulator * r,int v)84*4882a593Smuzhiyun int soc_pcmcia_regulator_set(struct soc_pcmcia_socket *skt,
85*4882a593Smuzhiyun 	struct soc_pcmcia_regulator *r, int v)
86*4882a593Smuzhiyun {
87*4882a593Smuzhiyun 	bool on;
88*4882a593Smuzhiyun 	int ret;
89*4882a593Smuzhiyun 
90*4882a593Smuzhiyun 	if (!r->reg)
91*4882a593Smuzhiyun 		return 0;
92*4882a593Smuzhiyun 
93*4882a593Smuzhiyun 	on = v != 0;
94*4882a593Smuzhiyun 	if (r->on == on)
95*4882a593Smuzhiyun 		return 0;
96*4882a593Smuzhiyun 
97*4882a593Smuzhiyun 	if (on) {
98*4882a593Smuzhiyun 		ret = regulator_set_voltage(r->reg, v * 100000, v * 100000);
99*4882a593Smuzhiyun 		if (ret) {
100*4882a593Smuzhiyun 			int vout = regulator_get_voltage(r->reg) / 100000;
101*4882a593Smuzhiyun 
102*4882a593Smuzhiyun 			dev_warn(&skt->socket.dev,
103*4882a593Smuzhiyun 				 "CS requested %s=%u.%uV, applying %u.%uV\n",
104*4882a593Smuzhiyun 				 r == &skt->vcc ? "Vcc" : "Vpp",
105*4882a593Smuzhiyun 				 v / 10, v % 10, vout / 10, vout % 10);
106*4882a593Smuzhiyun 		}
107*4882a593Smuzhiyun 
108*4882a593Smuzhiyun 		ret = regulator_enable(r->reg);
109*4882a593Smuzhiyun 	} else {
110*4882a593Smuzhiyun 		ret = regulator_disable(r->reg);
111*4882a593Smuzhiyun 	}
112*4882a593Smuzhiyun 	if (ret == 0)
113*4882a593Smuzhiyun 		r->on = on;
114*4882a593Smuzhiyun 
115*4882a593Smuzhiyun 	return ret;
116*4882a593Smuzhiyun }
117*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(soc_pcmcia_regulator_set);
118*4882a593Smuzhiyun 
119*4882a593Smuzhiyun static unsigned short
calc_speed(unsigned short * spds,int num,unsigned short dflt)120*4882a593Smuzhiyun calc_speed(unsigned short *spds, int num, unsigned short dflt)
121*4882a593Smuzhiyun {
122*4882a593Smuzhiyun 	unsigned short speed = 0;
123*4882a593Smuzhiyun 	int i;
124*4882a593Smuzhiyun 
125*4882a593Smuzhiyun 	for (i = 0; i < num; i++)
126*4882a593Smuzhiyun 		if (speed < spds[i])
127*4882a593Smuzhiyun 			speed = spds[i];
128*4882a593Smuzhiyun 	if (speed == 0)
129*4882a593Smuzhiyun 		speed = dflt;
130*4882a593Smuzhiyun 
131*4882a593Smuzhiyun 	return speed;
132*4882a593Smuzhiyun }
133*4882a593Smuzhiyun 
soc_common_pcmcia_get_timing(struct soc_pcmcia_socket * skt,struct soc_pcmcia_timing * timing)134*4882a593Smuzhiyun void soc_common_pcmcia_get_timing(struct soc_pcmcia_socket *skt,
135*4882a593Smuzhiyun 	struct soc_pcmcia_timing *timing)
136*4882a593Smuzhiyun {
137*4882a593Smuzhiyun 	timing->io =
138*4882a593Smuzhiyun 		calc_speed(skt->spd_io, MAX_IO_WIN, SOC_PCMCIA_IO_ACCESS);
139*4882a593Smuzhiyun 	timing->mem =
140*4882a593Smuzhiyun 		calc_speed(skt->spd_mem, MAX_WIN, SOC_PCMCIA_3V_MEM_ACCESS);
141*4882a593Smuzhiyun 	timing->attr =
142*4882a593Smuzhiyun 		calc_speed(skt->spd_attr, MAX_WIN, SOC_PCMCIA_3V_MEM_ACCESS);
143*4882a593Smuzhiyun }
144*4882a593Smuzhiyun EXPORT_SYMBOL(soc_common_pcmcia_get_timing);
145*4882a593Smuzhiyun 
__soc_pcmcia_hw_shutdown(struct soc_pcmcia_socket * skt,unsigned int nr)146*4882a593Smuzhiyun static void __soc_pcmcia_hw_shutdown(struct soc_pcmcia_socket *skt,
147*4882a593Smuzhiyun 	unsigned int nr)
148*4882a593Smuzhiyun {
149*4882a593Smuzhiyun 	unsigned int i;
150*4882a593Smuzhiyun 
151*4882a593Smuzhiyun 	for (i = 0; i < nr; i++)
152*4882a593Smuzhiyun 		if (skt->stat[i].irq)
153*4882a593Smuzhiyun 			free_irq(skt->stat[i].irq, skt);
154*4882a593Smuzhiyun 
155*4882a593Smuzhiyun 	if (skt->ops->hw_shutdown)
156*4882a593Smuzhiyun 		skt->ops->hw_shutdown(skt);
157*4882a593Smuzhiyun 
158*4882a593Smuzhiyun 	clk_disable_unprepare(skt->clk);
159*4882a593Smuzhiyun }
160*4882a593Smuzhiyun 
soc_pcmcia_hw_shutdown(struct soc_pcmcia_socket * skt)161*4882a593Smuzhiyun static void soc_pcmcia_hw_shutdown(struct soc_pcmcia_socket *skt)
162*4882a593Smuzhiyun {
163*4882a593Smuzhiyun 	__soc_pcmcia_hw_shutdown(skt, ARRAY_SIZE(skt->stat));
164*4882a593Smuzhiyun }
165*4882a593Smuzhiyun 
soc_pcmcia_request_gpiods(struct soc_pcmcia_socket * skt)166*4882a593Smuzhiyun int soc_pcmcia_request_gpiods(struct soc_pcmcia_socket *skt)
167*4882a593Smuzhiyun {
168*4882a593Smuzhiyun 	struct device *dev = skt->socket.dev.parent;
169*4882a593Smuzhiyun 	struct gpio_desc *desc;
170*4882a593Smuzhiyun 	int i;
171*4882a593Smuzhiyun 
172*4882a593Smuzhiyun 	for (i = 0; i < ARRAY_SIZE(skt->stat); i++) {
173*4882a593Smuzhiyun 		if (!skt->stat[i].name)
174*4882a593Smuzhiyun 			continue;
175*4882a593Smuzhiyun 
176*4882a593Smuzhiyun 		desc = devm_gpiod_get(dev, skt->stat[i].name, GPIOD_IN);
177*4882a593Smuzhiyun 		if (IS_ERR(desc)) {
178*4882a593Smuzhiyun 			dev_err(dev, "Failed to get GPIO for %s: %ld\n",
179*4882a593Smuzhiyun 				skt->stat[i].name, PTR_ERR(desc));
180*4882a593Smuzhiyun 			return PTR_ERR(desc);
181*4882a593Smuzhiyun 		}
182*4882a593Smuzhiyun 
183*4882a593Smuzhiyun 		skt->stat[i].desc = desc;
184*4882a593Smuzhiyun 	}
185*4882a593Smuzhiyun 
186*4882a593Smuzhiyun 	return 0;
187*4882a593Smuzhiyun }
188*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(soc_pcmcia_request_gpiods);
189*4882a593Smuzhiyun 
soc_pcmcia_hw_init(struct soc_pcmcia_socket * skt)190*4882a593Smuzhiyun static int soc_pcmcia_hw_init(struct soc_pcmcia_socket *skt)
191*4882a593Smuzhiyun {
192*4882a593Smuzhiyun 	int ret = 0, i;
193*4882a593Smuzhiyun 
194*4882a593Smuzhiyun 	ret = clk_prepare_enable(skt->clk);
195*4882a593Smuzhiyun 	if (ret)
196*4882a593Smuzhiyun 		return ret;
197*4882a593Smuzhiyun 
198*4882a593Smuzhiyun 	if (skt->ops->hw_init) {
199*4882a593Smuzhiyun 		ret = skt->ops->hw_init(skt);
200*4882a593Smuzhiyun 		if (ret) {
201*4882a593Smuzhiyun 			clk_disable_unprepare(skt->clk);
202*4882a593Smuzhiyun 			return ret;
203*4882a593Smuzhiyun 		}
204*4882a593Smuzhiyun 	}
205*4882a593Smuzhiyun 
206*4882a593Smuzhiyun 	for (i = 0; i < ARRAY_SIZE(skt->stat); i++) {
207*4882a593Smuzhiyun 		if (gpio_is_valid(skt->stat[i].gpio)) {
208*4882a593Smuzhiyun 			unsigned long flags = GPIOF_IN;
209*4882a593Smuzhiyun 
210*4882a593Smuzhiyun 			/* CD is active low by default */
211*4882a593Smuzhiyun 			if (i == SOC_STAT_CD)
212*4882a593Smuzhiyun 				flags |= GPIOF_ACTIVE_LOW;
213*4882a593Smuzhiyun 
214*4882a593Smuzhiyun 			ret = devm_gpio_request_one(skt->socket.dev.parent,
215*4882a593Smuzhiyun 						    skt->stat[i].gpio, flags,
216*4882a593Smuzhiyun 						    skt->stat[i].name);
217*4882a593Smuzhiyun 			if (ret) {
218*4882a593Smuzhiyun 				__soc_pcmcia_hw_shutdown(skt, i);
219*4882a593Smuzhiyun 				return ret;
220*4882a593Smuzhiyun 			}
221*4882a593Smuzhiyun 
222*4882a593Smuzhiyun 			skt->stat[i].desc = gpio_to_desc(skt->stat[i].gpio);
223*4882a593Smuzhiyun 		}
224*4882a593Smuzhiyun 
225*4882a593Smuzhiyun 		if (i < SOC_STAT_VS1 && skt->stat[i].desc) {
226*4882a593Smuzhiyun 			int irq = gpiod_to_irq(skt->stat[i].desc);
227*4882a593Smuzhiyun 
228*4882a593Smuzhiyun 			if (irq > 0) {
229*4882a593Smuzhiyun 				if (i == SOC_STAT_RDY)
230*4882a593Smuzhiyun 					skt->socket.pci_irq = irq;
231*4882a593Smuzhiyun 				else
232*4882a593Smuzhiyun 					skt->stat[i].irq = irq;
233*4882a593Smuzhiyun 			}
234*4882a593Smuzhiyun 		}
235*4882a593Smuzhiyun 
236*4882a593Smuzhiyun 		if (skt->stat[i].irq) {
237*4882a593Smuzhiyun 			ret = request_irq(skt->stat[i].irq,
238*4882a593Smuzhiyun 					  soc_common_pcmcia_interrupt,
239*4882a593Smuzhiyun 					  IRQF_TRIGGER_NONE,
240*4882a593Smuzhiyun 					  skt->stat[i].name, skt);
241*4882a593Smuzhiyun 			if (ret) {
242*4882a593Smuzhiyun 				__soc_pcmcia_hw_shutdown(skt, i);
243*4882a593Smuzhiyun 				return ret;
244*4882a593Smuzhiyun 			}
245*4882a593Smuzhiyun 		}
246*4882a593Smuzhiyun 	}
247*4882a593Smuzhiyun 
248*4882a593Smuzhiyun 	return ret;
249*4882a593Smuzhiyun }
250*4882a593Smuzhiyun 
soc_pcmcia_hw_enable(struct soc_pcmcia_socket * skt)251*4882a593Smuzhiyun static void soc_pcmcia_hw_enable(struct soc_pcmcia_socket *skt)
252*4882a593Smuzhiyun {
253*4882a593Smuzhiyun 	int i;
254*4882a593Smuzhiyun 
255*4882a593Smuzhiyun 	for (i = 0; i < ARRAY_SIZE(skt->stat); i++)
256*4882a593Smuzhiyun 		if (skt->stat[i].irq) {
257*4882a593Smuzhiyun 			irq_set_irq_type(skt->stat[i].irq, IRQ_TYPE_EDGE_RISING);
258*4882a593Smuzhiyun 			irq_set_irq_type(skt->stat[i].irq, IRQ_TYPE_EDGE_BOTH);
259*4882a593Smuzhiyun 		}
260*4882a593Smuzhiyun }
261*4882a593Smuzhiyun 
soc_pcmcia_hw_disable(struct soc_pcmcia_socket * skt)262*4882a593Smuzhiyun static void soc_pcmcia_hw_disable(struct soc_pcmcia_socket *skt)
263*4882a593Smuzhiyun {
264*4882a593Smuzhiyun 	int i;
265*4882a593Smuzhiyun 
266*4882a593Smuzhiyun 	for (i = 0; i < ARRAY_SIZE(skt->stat); i++)
267*4882a593Smuzhiyun 		if (skt->stat[i].irq)
268*4882a593Smuzhiyun 			irq_set_irq_type(skt->stat[i].irq, IRQ_TYPE_NONE);
269*4882a593Smuzhiyun }
270*4882a593Smuzhiyun 
271*4882a593Smuzhiyun /*
272*4882a593Smuzhiyun  * The CF 3.0 specification says that cards tie VS1 to ground and leave
273*4882a593Smuzhiyun  * VS2 open.  Many implementations do not wire up the VS signals, so we
274*4882a593Smuzhiyun  * provide hard-coded values as per the CF 3.0 spec.
275*4882a593Smuzhiyun  */
soc_common_cf_socket_state(struct soc_pcmcia_socket * skt,struct pcmcia_state * state)276*4882a593Smuzhiyun void soc_common_cf_socket_state(struct soc_pcmcia_socket *skt,
277*4882a593Smuzhiyun 	struct pcmcia_state *state)
278*4882a593Smuzhiyun {
279*4882a593Smuzhiyun 	state->vs_3v = 1;
280*4882a593Smuzhiyun }
281*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(soc_common_cf_socket_state);
282*4882a593Smuzhiyun 
soc_common_pcmcia_skt_state(struct soc_pcmcia_socket * skt)283*4882a593Smuzhiyun static unsigned int soc_common_pcmcia_skt_state(struct soc_pcmcia_socket *skt)
284*4882a593Smuzhiyun {
285*4882a593Smuzhiyun 	struct pcmcia_state state;
286*4882a593Smuzhiyun 	unsigned int stat;
287*4882a593Smuzhiyun 
288*4882a593Smuzhiyun 	memset(&state, 0, sizeof(struct pcmcia_state));
289*4882a593Smuzhiyun 
290*4882a593Smuzhiyun 	/* Make battery voltage state report 'good' */
291*4882a593Smuzhiyun 	state.bvd1 = 1;
292*4882a593Smuzhiyun 	state.bvd2 = 1;
293*4882a593Smuzhiyun 
294*4882a593Smuzhiyun 	if (skt->stat[SOC_STAT_CD].desc)
295*4882a593Smuzhiyun 		state.detect = !!gpiod_get_value(skt->stat[SOC_STAT_CD].desc);
296*4882a593Smuzhiyun 	if (skt->stat[SOC_STAT_RDY].desc)
297*4882a593Smuzhiyun 		state.ready = !!gpiod_get_value(skt->stat[SOC_STAT_RDY].desc);
298*4882a593Smuzhiyun 	if (skt->stat[SOC_STAT_BVD1].desc)
299*4882a593Smuzhiyun 		state.bvd1 = !!gpiod_get_value(skt->stat[SOC_STAT_BVD1].desc);
300*4882a593Smuzhiyun 	if (skt->stat[SOC_STAT_BVD2].desc)
301*4882a593Smuzhiyun 		state.bvd2 = !!gpiod_get_value(skt->stat[SOC_STAT_BVD2].desc);
302*4882a593Smuzhiyun 	if (skt->stat[SOC_STAT_VS1].desc)
303*4882a593Smuzhiyun 		state.vs_3v = !!gpiod_get_value(skt->stat[SOC_STAT_VS1].desc);
304*4882a593Smuzhiyun 	if (skt->stat[SOC_STAT_VS2].desc)
305*4882a593Smuzhiyun 		state.vs_Xv = !!gpiod_get_value(skt->stat[SOC_STAT_VS2].desc);
306*4882a593Smuzhiyun 
307*4882a593Smuzhiyun 	skt->ops->socket_state(skt, &state);
308*4882a593Smuzhiyun 
309*4882a593Smuzhiyun 	stat = state.detect  ? SS_DETECT : 0;
310*4882a593Smuzhiyun 	stat |= state.ready  ? SS_READY  : 0;
311*4882a593Smuzhiyun 	stat |= state.wrprot ? SS_WRPROT : 0;
312*4882a593Smuzhiyun 	stat |= state.vs_3v  ? SS_3VCARD : 0;
313*4882a593Smuzhiyun 	stat |= state.vs_Xv  ? SS_XVCARD : 0;
314*4882a593Smuzhiyun 
315*4882a593Smuzhiyun 	/* The power status of individual sockets is not available
316*4882a593Smuzhiyun 	 * explicitly from the hardware, so we just remember the state
317*4882a593Smuzhiyun 	 * and regurgitate it upon request:
318*4882a593Smuzhiyun 	 */
319*4882a593Smuzhiyun 	stat |= skt->cs_state.Vcc ? SS_POWERON : 0;
320*4882a593Smuzhiyun 
321*4882a593Smuzhiyun 	if (skt->cs_state.flags & SS_IOCARD)
322*4882a593Smuzhiyun 		stat |= state.bvd1 ? 0 : SS_STSCHG;
323*4882a593Smuzhiyun 	else {
324*4882a593Smuzhiyun 		if (state.bvd1 == 0)
325*4882a593Smuzhiyun 			stat |= SS_BATDEAD;
326*4882a593Smuzhiyun 		else if (state.bvd2 == 0)
327*4882a593Smuzhiyun 			stat |= SS_BATWARN;
328*4882a593Smuzhiyun 	}
329*4882a593Smuzhiyun 	return stat;
330*4882a593Smuzhiyun }
331*4882a593Smuzhiyun 
332*4882a593Smuzhiyun /*
333*4882a593Smuzhiyun  * soc_common_pcmcia_config_skt
334*4882a593Smuzhiyun  * ^^^^^^^^^^^^^^^^^^^^^^^^^^^^
335*4882a593Smuzhiyun  *
336*4882a593Smuzhiyun  * Convert PCMCIA socket state to our socket configure structure.
337*4882a593Smuzhiyun  */
soc_common_pcmcia_config_skt(struct soc_pcmcia_socket * skt,socket_state_t * state)338*4882a593Smuzhiyun static int soc_common_pcmcia_config_skt(
339*4882a593Smuzhiyun 	struct soc_pcmcia_socket *skt, socket_state_t *state)
340*4882a593Smuzhiyun {
341*4882a593Smuzhiyun 	int ret;
342*4882a593Smuzhiyun 
343*4882a593Smuzhiyun 	ret = skt->ops->configure_socket(skt, state);
344*4882a593Smuzhiyun 	if (ret < 0) {
345*4882a593Smuzhiyun 		pr_err("soc_common_pcmcia: unable to configure socket %d\n",
346*4882a593Smuzhiyun 		       skt->nr);
347*4882a593Smuzhiyun 		/* restore the previous state */
348*4882a593Smuzhiyun 		WARN_ON(skt->ops->configure_socket(skt, &skt->cs_state));
349*4882a593Smuzhiyun 		return ret;
350*4882a593Smuzhiyun 	}
351*4882a593Smuzhiyun 
352*4882a593Smuzhiyun 	if (ret == 0) {
353*4882a593Smuzhiyun 		struct gpio_desc *descs[2];
354*4882a593Smuzhiyun 		DECLARE_BITMAP(values, 2);
355*4882a593Smuzhiyun 		int n = 0;
356*4882a593Smuzhiyun 
357*4882a593Smuzhiyun 		if (skt->gpio_reset) {
358*4882a593Smuzhiyun 			descs[n] = skt->gpio_reset;
359*4882a593Smuzhiyun 			__assign_bit(n++, values, state->flags & SS_RESET);
360*4882a593Smuzhiyun 		}
361*4882a593Smuzhiyun 		if (skt->gpio_bus_enable) {
362*4882a593Smuzhiyun 			descs[n] = skt->gpio_bus_enable;
363*4882a593Smuzhiyun 			__assign_bit(n++, values, state->flags & SS_OUTPUT_ENA);
364*4882a593Smuzhiyun 		}
365*4882a593Smuzhiyun 
366*4882a593Smuzhiyun 		if (n)
367*4882a593Smuzhiyun 			gpiod_set_array_value_cansleep(n, descs, NULL, values);
368*4882a593Smuzhiyun 
369*4882a593Smuzhiyun 		/*
370*4882a593Smuzhiyun 		 * This really needs a better solution.  The IRQ
371*4882a593Smuzhiyun 		 * may or may not be claimed by the driver.
372*4882a593Smuzhiyun 		 */
373*4882a593Smuzhiyun 		if (skt->irq_state != 1 && state->io_irq) {
374*4882a593Smuzhiyun 			skt->irq_state = 1;
375*4882a593Smuzhiyun 			irq_set_irq_type(skt->socket.pci_irq,
376*4882a593Smuzhiyun 					 IRQ_TYPE_EDGE_FALLING);
377*4882a593Smuzhiyun 		} else if (skt->irq_state == 1 && state->io_irq == 0) {
378*4882a593Smuzhiyun 			skt->irq_state = 0;
379*4882a593Smuzhiyun 			irq_set_irq_type(skt->socket.pci_irq, IRQ_TYPE_NONE);
380*4882a593Smuzhiyun 		}
381*4882a593Smuzhiyun 
382*4882a593Smuzhiyun 		skt->cs_state = *state;
383*4882a593Smuzhiyun 	}
384*4882a593Smuzhiyun 
385*4882a593Smuzhiyun 	return ret;
386*4882a593Smuzhiyun }
387*4882a593Smuzhiyun 
388*4882a593Smuzhiyun /* soc_common_pcmcia_sock_init()
389*4882a593Smuzhiyun  * ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
390*4882a593Smuzhiyun  *
391*4882a593Smuzhiyun  * (Re-)Initialise the socket, turning on status interrupts
392*4882a593Smuzhiyun  * and PCMCIA bus.  This must wait for power to stabilise
393*4882a593Smuzhiyun  * so that the card status signals report correctly.
394*4882a593Smuzhiyun  *
395*4882a593Smuzhiyun  * Returns: 0
396*4882a593Smuzhiyun  */
soc_common_pcmcia_sock_init(struct pcmcia_socket * sock)397*4882a593Smuzhiyun static int soc_common_pcmcia_sock_init(struct pcmcia_socket *sock)
398*4882a593Smuzhiyun {
399*4882a593Smuzhiyun 	struct soc_pcmcia_socket *skt = to_soc_pcmcia_socket(sock);
400*4882a593Smuzhiyun 
401*4882a593Smuzhiyun 	debug(skt, 2, "initializing socket\n");
402*4882a593Smuzhiyun 	if (skt->ops->socket_init)
403*4882a593Smuzhiyun 		skt->ops->socket_init(skt);
404*4882a593Smuzhiyun 	soc_pcmcia_hw_enable(skt);
405*4882a593Smuzhiyun 	return 0;
406*4882a593Smuzhiyun }
407*4882a593Smuzhiyun 
408*4882a593Smuzhiyun 
409*4882a593Smuzhiyun /*
410*4882a593Smuzhiyun  * soc_common_pcmcia_suspend()
411*4882a593Smuzhiyun  * ^^^^^^^^^^^^^^^^^^^^^^^^^^^
412*4882a593Smuzhiyun  *
413*4882a593Smuzhiyun  * Remove power on the socket, disable IRQs from the card.
414*4882a593Smuzhiyun  * Turn off status interrupts, and disable the PCMCIA bus.
415*4882a593Smuzhiyun  *
416*4882a593Smuzhiyun  * Returns: 0
417*4882a593Smuzhiyun  */
soc_common_pcmcia_suspend(struct pcmcia_socket * sock)418*4882a593Smuzhiyun static int soc_common_pcmcia_suspend(struct pcmcia_socket *sock)
419*4882a593Smuzhiyun {
420*4882a593Smuzhiyun 	struct soc_pcmcia_socket *skt = to_soc_pcmcia_socket(sock);
421*4882a593Smuzhiyun 
422*4882a593Smuzhiyun 	debug(skt, 2, "suspending socket\n");
423*4882a593Smuzhiyun 
424*4882a593Smuzhiyun 	soc_pcmcia_hw_disable(skt);
425*4882a593Smuzhiyun 	if (skt->ops->socket_suspend)
426*4882a593Smuzhiyun 		skt->ops->socket_suspend(skt);
427*4882a593Smuzhiyun 
428*4882a593Smuzhiyun 	return 0;
429*4882a593Smuzhiyun }
430*4882a593Smuzhiyun 
431*4882a593Smuzhiyun static DEFINE_SPINLOCK(status_lock);
432*4882a593Smuzhiyun 
soc_common_check_status(struct soc_pcmcia_socket * skt)433*4882a593Smuzhiyun static void soc_common_check_status(struct soc_pcmcia_socket *skt)
434*4882a593Smuzhiyun {
435*4882a593Smuzhiyun 	unsigned int events;
436*4882a593Smuzhiyun 
437*4882a593Smuzhiyun 	debug(skt, 4, "entering PCMCIA monitoring thread\n");
438*4882a593Smuzhiyun 
439*4882a593Smuzhiyun 	do {
440*4882a593Smuzhiyun 		unsigned int status;
441*4882a593Smuzhiyun 		unsigned long flags;
442*4882a593Smuzhiyun 
443*4882a593Smuzhiyun 		status = soc_common_pcmcia_skt_state(skt);
444*4882a593Smuzhiyun 
445*4882a593Smuzhiyun 		spin_lock_irqsave(&status_lock, flags);
446*4882a593Smuzhiyun 		events = (status ^ skt->status) & skt->cs_state.csc_mask;
447*4882a593Smuzhiyun 		skt->status = status;
448*4882a593Smuzhiyun 		spin_unlock_irqrestore(&status_lock, flags);
449*4882a593Smuzhiyun 
450*4882a593Smuzhiyun 		debug(skt, 4, "events: %s%s%s%s%s%s\n",
451*4882a593Smuzhiyun 			events == 0         ? "<NONE>"   : "",
452*4882a593Smuzhiyun 			events & SS_DETECT  ? "DETECT "  : "",
453*4882a593Smuzhiyun 			events & SS_READY   ? "READY "   : "",
454*4882a593Smuzhiyun 			events & SS_BATDEAD ? "BATDEAD " : "",
455*4882a593Smuzhiyun 			events & SS_BATWARN ? "BATWARN " : "",
456*4882a593Smuzhiyun 			events & SS_STSCHG  ? "STSCHG "  : "");
457*4882a593Smuzhiyun 
458*4882a593Smuzhiyun 		if (events)
459*4882a593Smuzhiyun 			pcmcia_parse_events(&skt->socket, events);
460*4882a593Smuzhiyun 	} while (events);
461*4882a593Smuzhiyun }
462*4882a593Smuzhiyun 
463*4882a593Smuzhiyun /* Let's poll for events in addition to IRQs since IRQ only is unreliable... */
soc_common_pcmcia_poll_event(struct timer_list * t)464*4882a593Smuzhiyun static void soc_common_pcmcia_poll_event(struct timer_list *t)
465*4882a593Smuzhiyun {
466*4882a593Smuzhiyun 	struct soc_pcmcia_socket *skt = from_timer(skt, t, poll_timer);
467*4882a593Smuzhiyun 	debug(skt, 4, "polling for events\n");
468*4882a593Smuzhiyun 
469*4882a593Smuzhiyun 	mod_timer(&skt->poll_timer, jiffies + SOC_PCMCIA_POLL_PERIOD);
470*4882a593Smuzhiyun 
471*4882a593Smuzhiyun 	soc_common_check_status(skt);
472*4882a593Smuzhiyun }
473*4882a593Smuzhiyun 
474*4882a593Smuzhiyun 
475*4882a593Smuzhiyun /*
476*4882a593Smuzhiyun  * Service routine for socket driver interrupts (requested by the
477*4882a593Smuzhiyun  * low-level PCMCIA init() operation via soc_common_pcmcia_thread()).
478*4882a593Smuzhiyun  * The actual interrupt-servicing work is performed by
479*4882a593Smuzhiyun  * soc_common_pcmcia_thread(), largely because the Card Services event-
480*4882a593Smuzhiyun  * handling code performs scheduling operations which cannot be
481*4882a593Smuzhiyun  * executed from within an interrupt context.
482*4882a593Smuzhiyun  */
soc_common_pcmcia_interrupt(int irq,void * dev)483*4882a593Smuzhiyun static irqreturn_t soc_common_pcmcia_interrupt(int irq, void *dev)
484*4882a593Smuzhiyun {
485*4882a593Smuzhiyun 	struct soc_pcmcia_socket *skt = dev;
486*4882a593Smuzhiyun 
487*4882a593Smuzhiyun 	debug(skt, 3, "servicing IRQ %d\n", irq);
488*4882a593Smuzhiyun 
489*4882a593Smuzhiyun 	soc_common_check_status(skt);
490*4882a593Smuzhiyun 
491*4882a593Smuzhiyun 	return IRQ_HANDLED;
492*4882a593Smuzhiyun }
493*4882a593Smuzhiyun 
494*4882a593Smuzhiyun 
495*4882a593Smuzhiyun /*
496*4882a593Smuzhiyun  *  Implements the get_status() operation for the in-kernel PCMCIA
497*4882a593Smuzhiyun  * service (formerly SS_GetStatus in Card Services). Essentially just
498*4882a593Smuzhiyun  * fills in bits in `status' according to internal driver state or
499*4882a593Smuzhiyun  * the value of the voltage detect chipselect register.
500*4882a593Smuzhiyun  *
501*4882a593Smuzhiyun  * As a debugging note, during card startup, the PCMCIA core issues
502*4882a593Smuzhiyun  * three set_socket() commands in a row the first with RESET deasserted,
503*4882a593Smuzhiyun  * the second with RESET asserted, and the last with RESET deasserted
504*4882a593Smuzhiyun  * again. Following the third set_socket(), a get_status() command will
505*4882a593Smuzhiyun  * be issued. The kernel is looking for the SS_READY flag (see
506*4882a593Smuzhiyun  * setup_socket(), reset_socket(), and unreset_socket() in cs.c).
507*4882a593Smuzhiyun  *
508*4882a593Smuzhiyun  * Returns: 0
509*4882a593Smuzhiyun  */
510*4882a593Smuzhiyun static int
soc_common_pcmcia_get_status(struct pcmcia_socket * sock,unsigned int * status)511*4882a593Smuzhiyun soc_common_pcmcia_get_status(struct pcmcia_socket *sock, unsigned int *status)
512*4882a593Smuzhiyun {
513*4882a593Smuzhiyun 	struct soc_pcmcia_socket *skt = to_soc_pcmcia_socket(sock);
514*4882a593Smuzhiyun 
515*4882a593Smuzhiyun 	skt->status = soc_common_pcmcia_skt_state(skt);
516*4882a593Smuzhiyun 	*status = skt->status;
517*4882a593Smuzhiyun 
518*4882a593Smuzhiyun 	return 0;
519*4882a593Smuzhiyun }
520*4882a593Smuzhiyun 
521*4882a593Smuzhiyun 
522*4882a593Smuzhiyun /*
523*4882a593Smuzhiyun  * Implements the set_socket() operation for the in-kernel PCMCIA
524*4882a593Smuzhiyun  * service (formerly SS_SetSocket in Card Services). We more or
525*4882a593Smuzhiyun  * less punt all of this work and let the kernel handle the details
526*4882a593Smuzhiyun  * of power configuration, reset, &c. We also record the value of
527*4882a593Smuzhiyun  * `state' in order to regurgitate it to the PCMCIA core later.
528*4882a593Smuzhiyun  */
soc_common_pcmcia_set_socket(struct pcmcia_socket * sock,socket_state_t * state)529*4882a593Smuzhiyun static int soc_common_pcmcia_set_socket(
530*4882a593Smuzhiyun 	struct pcmcia_socket *sock, socket_state_t *state)
531*4882a593Smuzhiyun {
532*4882a593Smuzhiyun 	struct soc_pcmcia_socket *skt = to_soc_pcmcia_socket(sock);
533*4882a593Smuzhiyun 
534*4882a593Smuzhiyun 	debug(skt, 2, "mask: %s%s%s%s%s%s flags: %s%s%s%s%s%s Vcc %d Vpp %d irq %d\n",
535*4882a593Smuzhiyun 			(state->csc_mask == 0)		? "<NONE> " :	"",
536*4882a593Smuzhiyun 			(state->csc_mask & SS_DETECT)	? "DETECT " :	"",
537*4882a593Smuzhiyun 			(state->csc_mask & SS_READY)	? "READY " :	"",
538*4882a593Smuzhiyun 			(state->csc_mask & SS_BATDEAD)	? "BATDEAD " :	"",
539*4882a593Smuzhiyun 			(state->csc_mask & SS_BATWARN)	? "BATWARN " :	"",
540*4882a593Smuzhiyun 			(state->csc_mask & SS_STSCHG)	? "STSCHG " :	"",
541*4882a593Smuzhiyun 			(state->flags == 0)		? "<NONE> " :	"",
542*4882a593Smuzhiyun 			(state->flags & SS_PWR_AUTO)	? "PWR_AUTO " :	"",
543*4882a593Smuzhiyun 			(state->flags & SS_IOCARD)	? "IOCARD " :	"",
544*4882a593Smuzhiyun 			(state->flags & SS_RESET)	? "RESET " :	"",
545*4882a593Smuzhiyun 			(state->flags & SS_SPKR_ENA)	? "SPKR_ENA " :	"",
546*4882a593Smuzhiyun 			(state->flags & SS_OUTPUT_ENA)	? "OUTPUT_ENA " : "",
547*4882a593Smuzhiyun 			state->Vcc, state->Vpp, state->io_irq);
548*4882a593Smuzhiyun 
549*4882a593Smuzhiyun 	return soc_common_pcmcia_config_skt(skt, state);
550*4882a593Smuzhiyun }
551*4882a593Smuzhiyun 
552*4882a593Smuzhiyun 
553*4882a593Smuzhiyun /*
554*4882a593Smuzhiyun  * Implements the set_io_map() operation for the in-kernel PCMCIA
555*4882a593Smuzhiyun  * service (formerly SS_SetIOMap in Card Services). We configure
556*4882a593Smuzhiyun  * the map speed as requested, but override the address ranges
557*4882a593Smuzhiyun  * supplied by Card Services.
558*4882a593Smuzhiyun  *
559*4882a593Smuzhiyun  * Returns: 0 on success, -1 on error
560*4882a593Smuzhiyun  */
soc_common_pcmcia_set_io_map(struct pcmcia_socket * sock,struct pccard_io_map * map)561*4882a593Smuzhiyun static int soc_common_pcmcia_set_io_map(
562*4882a593Smuzhiyun 	struct pcmcia_socket *sock, struct pccard_io_map *map)
563*4882a593Smuzhiyun {
564*4882a593Smuzhiyun 	struct soc_pcmcia_socket *skt = to_soc_pcmcia_socket(sock);
565*4882a593Smuzhiyun 	unsigned short speed = map->speed;
566*4882a593Smuzhiyun 
567*4882a593Smuzhiyun 	debug(skt, 2, "map %u  speed %u start 0x%08llx stop 0x%08llx\n",
568*4882a593Smuzhiyun 		map->map, map->speed, (unsigned long long)map->start,
569*4882a593Smuzhiyun 		(unsigned long long)map->stop);
570*4882a593Smuzhiyun 	debug(skt, 2, "flags: %s%s%s%s%s%s%s%s\n",
571*4882a593Smuzhiyun 		(map->flags == 0)		? "<NONE>"	: "",
572*4882a593Smuzhiyun 		(map->flags & MAP_ACTIVE)	? "ACTIVE "	: "",
573*4882a593Smuzhiyun 		(map->flags & MAP_16BIT)	? "16BIT "	: "",
574*4882a593Smuzhiyun 		(map->flags & MAP_AUTOSZ)	? "AUTOSZ "	: "",
575*4882a593Smuzhiyun 		(map->flags & MAP_0WS)		? "0WS "	: "",
576*4882a593Smuzhiyun 		(map->flags & MAP_WRPROT)	? "WRPROT "	: "",
577*4882a593Smuzhiyun 		(map->flags & MAP_USE_WAIT)	? "USE_WAIT "	: "",
578*4882a593Smuzhiyun 		(map->flags & MAP_PREFETCH)	? "PREFETCH "	: "");
579*4882a593Smuzhiyun 
580*4882a593Smuzhiyun 	if (map->map >= MAX_IO_WIN) {
581*4882a593Smuzhiyun 		printk(KERN_ERR "%s(): map (%d) out of range\n", __func__,
582*4882a593Smuzhiyun 		       map->map);
583*4882a593Smuzhiyun 		return -1;
584*4882a593Smuzhiyun 	}
585*4882a593Smuzhiyun 
586*4882a593Smuzhiyun 	if (map->flags & MAP_ACTIVE) {
587*4882a593Smuzhiyun 		if (speed == 0)
588*4882a593Smuzhiyun 			speed = SOC_PCMCIA_IO_ACCESS;
589*4882a593Smuzhiyun 	} else {
590*4882a593Smuzhiyun 		speed = 0;
591*4882a593Smuzhiyun 	}
592*4882a593Smuzhiyun 
593*4882a593Smuzhiyun 	skt->spd_io[map->map] = speed;
594*4882a593Smuzhiyun 	skt->ops->set_timing(skt);
595*4882a593Smuzhiyun 
596*4882a593Smuzhiyun 	if (map->stop == 1)
597*4882a593Smuzhiyun 		map->stop = PAGE_SIZE-1;
598*4882a593Smuzhiyun 
599*4882a593Smuzhiyun 	map->stop -= map->start;
600*4882a593Smuzhiyun 	map->stop += skt->socket.io_offset;
601*4882a593Smuzhiyun 	map->start = skt->socket.io_offset;
602*4882a593Smuzhiyun 
603*4882a593Smuzhiyun 	return 0;
604*4882a593Smuzhiyun }
605*4882a593Smuzhiyun 
606*4882a593Smuzhiyun 
607*4882a593Smuzhiyun /*
608*4882a593Smuzhiyun  * Implements the set_mem_map() operation for the in-kernel PCMCIA
609*4882a593Smuzhiyun  * service (formerly SS_SetMemMap in Card Services). We configure
610*4882a593Smuzhiyun  * the map speed as requested, but override the address ranges
611*4882a593Smuzhiyun  * supplied by Card Services.
612*4882a593Smuzhiyun  *
613*4882a593Smuzhiyun  * Returns: 0 on success, -ERRNO on error
614*4882a593Smuzhiyun  */
soc_common_pcmcia_set_mem_map(struct pcmcia_socket * sock,struct pccard_mem_map * map)615*4882a593Smuzhiyun static int soc_common_pcmcia_set_mem_map(
616*4882a593Smuzhiyun 	struct pcmcia_socket *sock, struct pccard_mem_map *map)
617*4882a593Smuzhiyun {
618*4882a593Smuzhiyun 	struct soc_pcmcia_socket *skt = to_soc_pcmcia_socket(sock);
619*4882a593Smuzhiyun 	struct resource *res;
620*4882a593Smuzhiyun 	unsigned short speed = map->speed;
621*4882a593Smuzhiyun 
622*4882a593Smuzhiyun 	debug(skt, 2, "map %u speed %u card_start %08x\n",
623*4882a593Smuzhiyun 		map->map, map->speed, map->card_start);
624*4882a593Smuzhiyun 	debug(skt, 2, "flags: %s%s%s%s%s%s%s%s\n",
625*4882a593Smuzhiyun 		(map->flags == 0)		? "<NONE>"	: "",
626*4882a593Smuzhiyun 		(map->flags & MAP_ACTIVE)	? "ACTIVE "	: "",
627*4882a593Smuzhiyun 		(map->flags & MAP_16BIT)	? "16BIT "	: "",
628*4882a593Smuzhiyun 		(map->flags & MAP_AUTOSZ)	? "AUTOSZ "	: "",
629*4882a593Smuzhiyun 		(map->flags & MAP_0WS)		? "0WS "	: "",
630*4882a593Smuzhiyun 		(map->flags & MAP_WRPROT)	? "WRPROT "	: "",
631*4882a593Smuzhiyun 		(map->flags & MAP_ATTRIB)	? "ATTRIB "	: "",
632*4882a593Smuzhiyun 		(map->flags & MAP_USE_WAIT)	? "USE_WAIT "	: "");
633*4882a593Smuzhiyun 
634*4882a593Smuzhiyun 	if (map->map >= MAX_WIN)
635*4882a593Smuzhiyun 		return -EINVAL;
636*4882a593Smuzhiyun 
637*4882a593Smuzhiyun 	if (map->flags & MAP_ACTIVE) {
638*4882a593Smuzhiyun 		if (speed == 0)
639*4882a593Smuzhiyun 			speed = 300;
640*4882a593Smuzhiyun 	} else {
641*4882a593Smuzhiyun 		speed = 0;
642*4882a593Smuzhiyun 	}
643*4882a593Smuzhiyun 
644*4882a593Smuzhiyun 	if (map->flags & MAP_ATTRIB) {
645*4882a593Smuzhiyun 		res = &skt->res_attr;
646*4882a593Smuzhiyun 		skt->spd_attr[map->map] = speed;
647*4882a593Smuzhiyun 		skt->spd_mem[map->map] = 0;
648*4882a593Smuzhiyun 	} else {
649*4882a593Smuzhiyun 		res = &skt->res_mem;
650*4882a593Smuzhiyun 		skt->spd_attr[map->map] = 0;
651*4882a593Smuzhiyun 		skt->spd_mem[map->map] = speed;
652*4882a593Smuzhiyun 	}
653*4882a593Smuzhiyun 
654*4882a593Smuzhiyun 	skt->ops->set_timing(skt);
655*4882a593Smuzhiyun 
656*4882a593Smuzhiyun 	map->static_start = res->start + map->card_start;
657*4882a593Smuzhiyun 
658*4882a593Smuzhiyun 	return 0;
659*4882a593Smuzhiyun }
660*4882a593Smuzhiyun 
661*4882a593Smuzhiyun struct bittbl {
662*4882a593Smuzhiyun 	unsigned int mask;
663*4882a593Smuzhiyun 	const char *name;
664*4882a593Smuzhiyun };
665*4882a593Smuzhiyun 
666*4882a593Smuzhiyun static struct bittbl status_bits[] = {
667*4882a593Smuzhiyun 	{ SS_WRPROT,		"SS_WRPROT"	},
668*4882a593Smuzhiyun 	{ SS_BATDEAD,		"SS_BATDEAD"	},
669*4882a593Smuzhiyun 	{ SS_BATWARN,		"SS_BATWARN"	},
670*4882a593Smuzhiyun 	{ SS_READY,		"SS_READY"	},
671*4882a593Smuzhiyun 	{ SS_DETECT,		"SS_DETECT"	},
672*4882a593Smuzhiyun 	{ SS_POWERON,		"SS_POWERON"	},
673*4882a593Smuzhiyun 	{ SS_STSCHG,		"SS_STSCHG"	},
674*4882a593Smuzhiyun 	{ SS_3VCARD,		"SS_3VCARD"	},
675*4882a593Smuzhiyun 	{ SS_XVCARD,		"SS_XVCARD"	},
676*4882a593Smuzhiyun };
677*4882a593Smuzhiyun 
678*4882a593Smuzhiyun static struct bittbl conf_bits[] = {
679*4882a593Smuzhiyun 	{ SS_PWR_AUTO,		"SS_PWR_AUTO"	},
680*4882a593Smuzhiyun 	{ SS_IOCARD,		"SS_IOCARD"	},
681*4882a593Smuzhiyun 	{ SS_RESET,		"SS_RESET"	},
682*4882a593Smuzhiyun 	{ SS_DMA_MODE,		"SS_DMA_MODE"	},
683*4882a593Smuzhiyun 	{ SS_SPKR_ENA,		"SS_SPKR_ENA"	},
684*4882a593Smuzhiyun 	{ SS_OUTPUT_ENA,	"SS_OUTPUT_ENA"	},
685*4882a593Smuzhiyun };
686*4882a593Smuzhiyun 
dump_bits(char ** p,const char * prefix,unsigned int val,struct bittbl * bits,int sz)687*4882a593Smuzhiyun static void dump_bits(char **p, const char *prefix,
688*4882a593Smuzhiyun 	unsigned int val, struct bittbl *bits, int sz)
689*4882a593Smuzhiyun {
690*4882a593Smuzhiyun 	char *b = *p;
691*4882a593Smuzhiyun 	int i;
692*4882a593Smuzhiyun 
693*4882a593Smuzhiyun 	b += sprintf(b, "%-9s:", prefix);
694*4882a593Smuzhiyun 	for (i = 0; i < sz; i++)
695*4882a593Smuzhiyun 		if (val & bits[i].mask)
696*4882a593Smuzhiyun 			b += sprintf(b, " %s", bits[i].name);
697*4882a593Smuzhiyun 	*b++ = '\n';
698*4882a593Smuzhiyun 	*p = b;
699*4882a593Smuzhiyun }
700*4882a593Smuzhiyun 
701*4882a593Smuzhiyun /*
702*4882a593Smuzhiyun  * Implements the /sys/class/pcmcia_socket/??/status file.
703*4882a593Smuzhiyun  *
704*4882a593Smuzhiyun  * Returns: the number of characters added to the buffer
705*4882a593Smuzhiyun  */
show_status(struct device * dev,struct device_attribute * attr,char * buf)706*4882a593Smuzhiyun static ssize_t show_status(
707*4882a593Smuzhiyun 	struct device *dev, struct device_attribute *attr, char *buf)
708*4882a593Smuzhiyun {
709*4882a593Smuzhiyun 	struct soc_pcmcia_socket *skt =
710*4882a593Smuzhiyun 		container_of(dev, struct soc_pcmcia_socket, socket.dev);
711*4882a593Smuzhiyun 	char *p = buf;
712*4882a593Smuzhiyun 
713*4882a593Smuzhiyun 	p += sprintf(p, "slot     : %d\n", skt->nr);
714*4882a593Smuzhiyun 
715*4882a593Smuzhiyun 	dump_bits(&p, "status", skt->status,
716*4882a593Smuzhiyun 		  status_bits, ARRAY_SIZE(status_bits));
717*4882a593Smuzhiyun 	dump_bits(&p, "csc_mask", skt->cs_state.csc_mask,
718*4882a593Smuzhiyun 		  status_bits, ARRAY_SIZE(status_bits));
719*4882a593Smuzhiyun 	dump_bits(&p, "cs_flags", skt->cs_state.flags,
720*4882a593Smuzhiyun 		  conf_bits, ARRAY_SIZE(conf_bits));
721*4882a593Smuzhiyun 
722*4882a593Smuzhiyun 	p += sprintf(p, "Vcc      : %d\n", skt->cs_state.Vcc);
723*4882a593Smuzhiyun 	p += sprintf(p, "Vpp      : %d\n", skt->cs_state.Vpp);
724*4882a593Smuzhiyun 	p += sprintf(p, "IRQ      : %d (%d)\n", skt->cs_state.io_irq,
725*4882a593Smuzhiyun 		skt->socket.pci_irq);
726*4882a593Smuzhiyun 	if (skt->ops->show_timing)
727*4882a593Smuzhiyun 		p += skt->ops->show_timing(skt, p);
728*4882a593Smuzhiyun 
729*4882a593Smuzhiyun 	return p-buf;
730*4882a593Smuzhiyun }
731*4882a593Smuzhiyun static DEVICE_ATTR(status, S_IRUGO, show_status, NULL);
732*4882a593Smuzhiyun 
733*4882a593Smuzhiyun 
734*4882a593Smuzhiyun static struct pccard_operations soc_common_pcmcia_operations = {
735*4882a593Smuzhiyun 	.init			= soc_common_pcmcia_sock_init,
736*4882a593Smuzhiyun 	.suspend		= soc_common_pcmcia_suspend,
737*4882a593Smuzhiyun 	.get_status		= soc_common_pcmcia_get_status,
738*4882a593Smuzhiyun 	.set_socket		= soc_common_pcmcia_set_socket,
739*4882a593Smuzhiyun 	.set_io_map		= soc_common_pcmcia_set_io_map,
740*4882a593Smuzhiyun 	.set_mem_map		= soc_common_pcmcia_set_mem_map,
741*4882a593Smuzhiyun };
742*4882a593Smuzhiyun 
743*4882a593Smuzhiyun 
744*4882a593Smuzhiyun #ifdef CONFIG_CPU_FREQ
soc_common_pcmcia_cpufreq_nb(struct notifier_block * nb,unsigned long val,void * data)745*4882a593Smuzhiyun static int soc_common_pcmcia_cpufreq_nb(struct notifier_block *nb,
746*4882a593Smuzhiyun 	unsigned long val, void *data)
747*4882a593Smuzhiyun {
748*4882a593Smuzhiyun 	struct soc_pcmcia_socket *skt = container_of(nb, struct soc_pcmcia_socket, cpufreq_nb);
749*4882a593Smuzhiyun 	struct cpufreq_freqs *freqs = data;
750*4882a593Smuzhiyun 
751*4882a593Smuzhiyun 	return skt->ops->frequency_change(skt, val, freqs);
752*4882a593Smuzhiyun }
753*4882a593Smuzhiyun #endif
754*4882a593Smuzhiyun 
soc_pcmcia_init_one(struct soc_pcmcia_socket * skt,const struct pcmcia_low_level * ops,struct device * dev)755*4882a593Smuzhiyun void soc_pcmcia_init_one(struct soc_pcmcia_socket *skt,
756*4882a593Smuzhiyun 	const struct pcmcia_low_level *ops, struct device *dev)
757*4882a593Smuzhiyun {
758*4882a593Smuzhiyun 	int i;
759*4882a593Smuzhiyun 
760*4882a593Smuzhiyun 	skt->ops = ops;
761*4882a593Smuzhiyun 	skt->socket.owner = ops->owner;
762*4882a593Smuzhiyun 	skt->socket.dev.parent = dev;
763*4882a593Smuzhiyun 	skt->socket.pci_irq = NO_IRQ;
764*4882a593Smuzhiyun 
765*4882a593Smuzhiyun 	for (i = 0; i < ARRAY_SIZE(skt->stat); i++)
766*4882a593Smuzhiyun 		skt->stat[i].gpio = -EINVAL;
767*4882a593Smuzhiyun }
768*4882a593Smuzhiyun EXPORT_SYMBOL(soc_pcmcia_init_one);
769*4882a593Smuzhiyun 
soc_pcmcia_remove_one(struct soc_pcmcia_socket * skt)770*4882a593Smuzhiyun void soc_pcmcia_remove_one(struct soc_pcmcia_socket *skt)
771*4882a593Smuzhiyun {
772*4882a593Smuzhiyun 	del_timer_sync(&skt->poll_timer);
773*4882a593Smuzhiyun 
774*4882a593Smuzhiyun 	pcmcia_unregister_socket(&skt->socket);
775*4882a593Smuzhiyun 
776*4882a593Smuzhiyun #ifdef CONFIG_CPU_FREQ
777*4882a593Smuzhiyun 	if (skt->ops->frequency_change)
778*4882a593Smuzhiyun 		cpufreq_unregister_notifier(&skt->cpufreq_nb,
779*4882a593Smuzhiyun 					    CPUFREQ_TRANSITION_NOTIFIER);
780*4882a593Smuzhiyun #endif
781*4882a593Smuzhiyun 
782*4882a593Smuzhiyun 	soc_pcmcia_hw_shutdown(skt);
783*4882a593Smuzhiyun 
784*4882a593Smuzhiyun 	/* should not be required; violates some lowlevel drivers */
785*4882a593Smuzhiyun 	soc_common_pcmcia_config_skt(skt, &dead_socket);
786*4882a593Smuzhiyun 
787*4882a593Smuzhiyun 	iounmap(skt->virt_io);
788*4882a593Smuzhiyun 	skt->virt_io = NULL;
789*4882a593Smuzhiyun 	release_resource(&skt->res_attr);
790*4882a593Smuzhiyun 	release_resource(&skt->res_mem);
791*4882a593Smuzhiyun 	release_resource(&skt->res_io);
792*4882a593Smuzhiyun 	release_resource(&skt->res_skt);
793*4882a593Smuzhiyun }
794*4882a593Smuzhiyun EXPORT_SYMBOL(soc_pcmcia_remove_one);
795*4882a593Smuzhiyun 
soc_pcmcia_add_one(struct soc_pcmcia_socket * skt)796*4882a593Smuzhiyun int soc_pcmcia_add_one(struct soc_pcmcia_socket *skt)
797*4882a593Smuzhiyun {
798*4882a593Smuzhiyun 	int ret;
799*4882a593Smuzhiyun 
800*4882a593Smuzhiyun 	skt->cs_state = dead_socket;
801*4882a593Smuzhiyun 
802*4882a593Smuzhiyun 	timer_setup(&skt->poll_timer, soc_common_pcmcia_poll_event, 0);
803*4882a593Smuzhiyun 	skt->poll_timer.expires = jiffies + SOC_PCMCIA_POLL_PERIOD;
804*4882a593Smuzhiyun 
805*4882a593Smuzhiyun 	ret = request_resource(&iomem_resource, &skt->res_skt);
806*4882a593Smuzhiyun 	if (ret)
807*4882a593Smuzhiyun 		goto out_err_1;
808*4882a593Smuzhiyun 
809*4882a593Smuzhiyun 	ret = request_resource(&skt->res_skt, &skt->res_io);
810*4882a593Smuzhiyun 	if (ret)
811*4882a593Smuzhiyun 		goto out_err_2;
812*4882a593Smuzhiyun 
813*4882a593Smuzhiyun 	ret = request_resource(&skt->res_skt, &skt->res_mem);
814*4882a593Smuzhiyun 	if (ret)
815*4882a593Smuzhiyun 		goto out_err_3;
816*4882a593Smuzhiyun 
817*4882a593Smuzhiyun 	ret = request_resource(&skt->res_skt, &skt->res_attr);
818*4882a593Smuzhiyun 	if (ret)
819*4882a593Smuzhiyun 		goto out_err_4;
820*4882a593Smuzhiyun 
821*4882a593Smuzhiyun 	skt->virt_io = ioremap(skt->res_io.start, 0x10000);
822*4882a593Smuzhiyun 	if (skt->virt_io == NULL) {
823*4882a593Smuzhiyun 		ret = -ENOMEM;
824*4882a593Smuzhiyun 		goto out_err_5;
825*4882a593Smuzhiyun 	}
826*4882a593Smuzhiyun 
827*4882a593Smuzhiyun 	/*
828*4882a593Smuzhiyun 	 * We initialize default socket timing here, because
829*4882a593Smuzhiyun 	 * we are not guaranteed to see a SetIOMap operation at
830*4882a593Smuzhiyun 	 * runtime.
831*4882a593Smuzhiyun 	 */
832*4882a593Smuzhiyun 	skt->ops->set_timing(skt);
833*4882a593Smuzhiyun 
834*4882a593Smuzhiyun 	ret = soc_pcmcia_hw_init(skt);
835*4882a593Smuzhiyun 	if (ret)
836*4882a593Smuzhiyun 		goto out_err_6;
837*4882a593Smuzhiyun 
838*4882a593Smuzhiyun 	skt->socket.ops = &soc_common_pcmcia_operations;
839*4882a593Smuzhiyun 	skt->socket.features = SS_CAP_STATIC_MAP|SS_CAP_PCCARD;
840*4882a593Smuzhiyun 	skt->socket.resource_ops = &pccard_static_ops;
841*4882a593Smuzhiyun 	skt->socket.irq_mask = 0;
842*4882a593Smuzhiyun 	skt->socket.map_size = PAGE_SIZE;
843*4882a593Smuzhiyun 	skt->socket.io_offset = (unsigned long)skt->virt_io;
844*4882a593Smuzhiyun 
845*4882a593Smuzhiyun 	skt->status = soc_common_pcmcia_skt_state(skt);
846*4882a593Smuzhiyun 
847*4882a593Smuzhiyun #ifdef CONFIG_CPU_FREQ
848*4882a593Smuzhiyun 	if (skt->ops->frequency_change) {
849*4882a593Smuzhiyun 		skt->cpufreq_nb.notifier_call = soc_common_pcmcia_cpufreq_nb;
850*4882a593Smuzhiyun 
851*4882a593Smuzhiyun 		ret = cpufreq_register_notifier(&skt->cpufreq_nb,
852*4882a593Smuzhiyun 						CPUFREQ_TRANSITION_NOTIFIER);
853*4882a593Smuzhiyun 		if (ret < 0)
854*4882a593Smuzhiyun 			dev_err(skt->socket.dev.parent,
855*4882a593Smuzhiyun 				"unable to register CPU frequency change notifier for PCMCIA (%d)\n",
856*4882a593Smuzhiyun 				ret);
857*4882a593Smuzhiyun 	}
858*4882a593Smuzhiyun #endif
859*4882a593Smuzhiyun 
860*4882a593Smuzhiyun 	ret = pcmcia_register_socket(&skt->socket);
861*4882a593Smuzhiyun 	if (ret)
862*4882a593Smuzhiyun 		goto out_err_7;
863*4882a593Smuzhiyun 
864*4882a593Smuzhiyun 	ret = device_create_file(&skt->socket.dev, &dev_attr_status);
865*4882a593Smuzhiyun 	if (ret)
866*4882a593Smuzhiyun 		goto out_err_8;
867*4882a593Smuzhiyun 
868*4882a593Smuzhiyun 	return ret;
869*4882a593Smuzhiyun 
870*4882a593Smuzhiyun  out_err_8:
871*4882a593Smuzhiyun 	del_timer_sync(&skt->poll_timer);
872*4882a593Smuzhiyun 	pcmcia_unregister_socket(&skt->socket);
873*4882a593Smuzhiyun 
874*4882a593Smuzhiyun  out_err_7:
875*4882a593Smuzhiyun 	soc_pcmcia_hw_shutdown(skt);
876*4882a593Smuzhiyun  out_err_6:
877*4882a593Smuzhiyun 	iounmap(skt->virt_io);
878*4882a593Smuzhiyun  out_err_5:
879*4882a593Smuzhiyun 	release_resource(&skt->res_attr);
880*4882a593Smuzhiyun  out_err_4:
881*4882a593Smuzhiyun 	release_resource(&skt->res_mem);
882*4882a593Smuzhiyun  out_err_3:
883*4882a593Smuzhiyun 	release_resource(&skt->res_io);
884*4882a593Smuzhiyun  out_err_2:
885*4882a593Smuzhiyun 	release_resource(&skt->res_skt);
886*4882a593Smuzhiyun  out_err_1:
887*4882a593Smuzhiyun 
888*4882a593Smuzhiyun 	return ret;
889*4882a593Smuzhiyun }
890*4882a593Smuzhiyun EXPORT_SYMBOL(soc_pcmcia_add_one);
891*4882a593Smuzhiyun 
892*4882a593Smuzhiyun MODULE_AUTHOR("John Dorsey <john+@cs.cmu.edu>");
893*4882a593Smuzhiyun MODULE_DESCRIPTION("Linux PCMCIA Card Services: Common SoC support");
894*4882a593Smuzhiyun MODULE_LICENSE("Dual MPL/GPL");
895