2 * GTA02 WLAN power management
4 * (C) 2008, 2009 by Openmoko Inc.
5 * Author: Andy Green <andy@openmoko.com>
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation
14 #include <linux/module.h>
15 #include <linux/init.h>
16 #include <linux/kernel.h>
17 #include <linux/mutex.h>
18 #include <linux/platform_device.h>
20 #include <mach/hardware.h>
21 #include <asm/mach-types.h>
23 #include <mach/gta02.h>
24 #include <mach/gta02-pm-wlan.h>
25 #include <mach/regs-gpio.h>
26 #include <mach/regs-gpioj.h>
28 #include <linux/delay.h>
29 #include <linux/rfkill.h>
32 /* ----- Module hardware reset ("power") ----------------------------------- */
35 void gta02_wlan_reset(int assert_reset
)
38 s3c2410_gpio_setpin(GTA02_GPIO_nWLAN_RESET
, 0);
39 msleep(200); /* probably excessive but we don't have specs */
41 s3c2410_gpio_setpin(GTA02_GPIO_nWLAN_RESET
, 1);
45 /* ----- rfkill ------------------------------------------------------------ */
48 * S3C MCI handles suspend/resume through device removal/insertion. In order to
49 * preserve rfkill state, as required in clause 7 of section 3.1 in rfkill.txt,
50 * we therefore need to maintain rfkill state outside the driver.
52 * This platform driver is as good a place as any other.
55 static int (*gta02_wlan_rfkill_cb
)(void *user
, int on
);
56 static void *gta02_wlan_rfkill_user
;
57 static DEFINE_MUTEX(gta02_wlan_rfkill_lock
);
58 static int gta02_wlan_rfkill_on
;
61 * gta02_wlan_query_rfkill_lock is used to obtain the rfkill state before the
62 * driver is ready to process rfkill callbacks. To prevent the state from
63 * changing until the driver has completed its initialization, we grab and hold
66 * A call to gta02_wlan_query_rfkill_lock must be followed by either
67 * - a call to gta02_wlan_set_rfkill_cb, to complete the setup, or
68 * - a call to gta02_wlan_query_rfkill_unlock to abort the setup process.
71 int gta02_wlan_query_rfkill_lock(void)
73 mutex_lock(>a02_wlan_rfkill_lock
);
74 return gta02_wlan_rfkill_on
;
76 EXPORT_SYMBOL_GPL(gta02_wlan_query_rfkill_lock
);
78 void gta02_wlan_query_rfkill_unlock(void)
80 mutex_unlock(>a02_wlan_rfkill_lock
);
82 EXPORT_SYMBOL_GPL(gta02_wlan_query_rfkill_unlock
);
84 void gta02_wlan_set_rfkill_cb(int (*cb
)(void *user
, int on
), void *user
)
86 BUG_ON(!mutex_is_locked(>a02_wlan_rfkill_lock
));
87 BUG_ON(gta02_wlan_rfkill_cb
);
88 gta02_wlan_rfkill_cb
= cb
;
89 gta02_wlan_rfkill_user
= user
;
90 mutex_unlock(>a02_wlan_rfkill_lock
);
92 EXPORT_SYMBOL_GPL(gta02_wlan_set_rfkill_cb
);
94 void gta02_wlan_clear_rfkill_cb(void)
96 mutex_lock(>a02_wlan_rfkill_lock
);
97 BUG_ON(!gta02_wlan_rfkill_cb
);
98 gta02_wlan_rfkill_cb
= NULL
;
99 mutex_unlock(>a02_wlan_rfkill_lock
);
101 EXPORT_SYMBOL_GPL(gta02_wlan_clear_rfkill_cb
);
103 static int gta02_wlan_toggle_radio(void *data
, enum rfkill_state state
)
105 struct device
*dev
= data
;
106 int on
= state
== RFKILL_STATE_UNBLOCKED
;
109 dev_dbg(dev
, "gta02_wlan_toggle_radio: state %d (%p)\n",
110 state
, gta02_wlan_rfkill_cb
);
111 mutex_lock(>a02_wlan_rfkill_lock
);
112 if (gta02_wlan_rfkill_cb
)
113 res
= gta02_wlan_rfkill_cb(gta02_wlan_rfkill_user
, on
);
115 gta02_wlan_rfkill_on
= on
;
116 mutex_unlock(>a02_wlan_rfkill_lock
);
121 /* ----- Initialization/removal -------------------------------------------- */
124 static int __init
gta02_wlan_probe(struct platform_device
*pdev
)
126 /* default-on for now */
127 const int default_state
= 1;
128 struct rfkill
*rfkill
;
131 dev_info(&pdev
->dev
, "starting\n");
133 s3c2410_gpio_cfgpin(GTA02_GPIO_nWLAN_RESET
, S3C2410_GPIO_OUTPUT
);
137 rfkill
= rfkill_allocate(&pdev
->dev
, RFKILL_TYPE_WLAN
);
138 rfkill
->name
= "ar6000";
139 rfkill
->data
= &pdev
->dev
;
140 rfkill
->state
= default_state
? RFKILL_STATE_ON
: RFKILL_STATE_OFF
;
142 * If the WLAN driver somehow managed to get activated before we're
143 * ready, the driver is now in an unknown state, which isn't something
144 * we're prepared to handle. This can't happen, so just fail hard.
146 BUG_ON(gta02_wlan_rfkill_cb
);
147 gta02_wlan_rfkill_on
= default_state
;
148 rfkill
->toggle_radio
= gta02_wlan_toggle_radio
;
150 error
= rfkill_register(rfkill
);
156 dev_set_drvdata(&pdev
->dev
, rfkill
);
161 static int gta02_wlan_remove(struct platform_device
*pdev
)
163 struct rfkill
*rfkill
= dev_get_drvdata(&pdev
->dev
);
165 rfkill_unregister(rfkill
);
171 static struct platform_driver gta02_wlan_driver
= {
172 .probe
= gta02_wlan_probe
,
173 .remove
= gta02_wlan_remove
,
175 .name
= "gta02-pm-wlan",
179 static int __devinit
gta02_wlan_init(void)
181 return platform_driver_register(>a02_wlan_driver
);
184 static void gta02_wlan_exit(void)
186 platform_driver_unregister(>a02_wlan_driver
);
189 module_init(gta02_wlan_init
);
190 module_exit(gta02_wlan_exit
);
192 MODULE_LICENSE("GPL");
193 MODULE_AUTHOR("Andy Green <andy@openmoko.com>");
194 MODULE_DESCRIPTION("Openmoko GTA02 WLAN power management");