Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 181836
b: refs/heads/master
c: 493e914
h: refs/heads/master
v: v3
  • Loading branch information
Mario Limonciello authored and Matthew Garrett committed Feb 25, 2010
1 parent 069888a commit da305af
Show file tree
Hide file tree
Showing 2 changed files with 64 additions and 139 deletions.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: 34325b9d2a7d36e2ba74a6652f2e3e8d57dfb145
refs/heads/master: 493e91433e966a50964e221db92756eeb90dd54e
201 changes: 63 additions & 138 deletions trunk/drivers/platform/x86/compal-laptop.c
Original file line number Diff line number Diff line change
Expand Up @@ -26,17 +26,8 @@
/*
* comapl-laptop.c - Compal laptop support.
*
* This driver exports a few files in /sys/devices/platform/compal-laptop/:
*
* wlan - wlan subsystem state: contains 0 or 1 (rw)
*
* bluetooth - Bluetooth subsystem state: contains 0 or 1 (rw)
*
* raw - raw value taken from embedded controller register (ro)
*
* In addition to these platform device attributes the driver
* registers itself in the Linux backlight control subsystem and is
* available to userspace under /sys/class/backlight/compal-laptop/.
* The driver registers itself with the rfkill subsystem and
* the Linux backlight control subsystem.
*
* This driver might work on other laptops produced by Compal. If you
* want to try it you can pass force=1 as argument to the module which
Expand All @@ -51,6 +42,7 @@
#include <linux/dmi.h>
#include <linux/backlight.h>
#include <linux/platform_device.h>
#include <linux/rfkill.h>

#define COMPAL_DRIVER_VERSION "0.2.6"

Expand All @@ -63,6 +55,10 @@
#define WLAN_MASK 0x01
#define BT_MASK 0x02

static struct rfkill *wifi_rfkill;
static struct rfkill *bt_rfkill;
static struct platform_device *compal_device;

static int force;
module_param(force, bool, 0);
MODULE_PARM_DESC(force, "Force driver load, ignore DMI data");
Expand All @@ -88,65 +84,75 @@ static int get_lcd_level(void)
return (int) result;
}

static int set_wlan_state(int state)
static int compal_rfkill_set(void *data, bool blocked)
{
unsigned long radio = (unsigned long) data;
u8 result, value;

ec_read(COMPAL_EC_COMMAND_WIRELESS, &result);

if ((result & KILLSWITCH_MASK) == 0)
return -EINVAL;
else {
if (state)
value = (u8) (result | WLAN_MASK);
else
value = (u8) (result & ~WLAN_MASK);
ec_write(COMPAL_EC_COMMAND_WIRELESS, value);
}
if (!blocked)
value = (u8) (result | radio);
else
value = (u8) (result & ~radio);
ec_write(COMPAL_EC_COMMAND_WIRELESS, value);

return 0;
}

static int set_bluetooth_state(int state)
static void compal_rfkill_poll(struct rfkill *rfkill, void *data)
{
u8 result, value;
u8 result;
bool hw_blocked;

ec_read(COMPAL_EC_COMMAND_WIRELESS, &result);

if ((result & KILLSWITCH_MASK) == 0)
return -EINVAL;
else {
if (state)
value = (u8) (result | BT_MASK);
else
value = (u8) (result & ~BT_MASK);
ec_write(COMPAL_EC_COMMAND_WIRELESS, value);
}

return 0;
hw_blocked = !(result & KILLSWITCH_MASK);
rfkill_set_hw_state(rfkill, hw_blocked);
}

static int get_wireless_state(int *wlan, int *bluetooth)
static const struct rfkill_ops compal_rfkill_ops = {
.poll = compal_rfkill_poll,
.set_block = compal_rfkill_set,
};

static int setup_rfkill(void)
{
u8 result;
int ret;

ec_read(COMPAL_EC_COMMAND_WIRELESS, &result);
wifi_rfkill = rfkill_alloc("compal-wifi", &compal_device->dev,
RFKILL_TYPE_WLAN, &compal_rfkill_ops,
(void *) WLAN_MASK);
if (!wifi_rfkill)
return -ENOMEM;

if (wlan) {
if ((result & KILLSWITCH_MASK) == 0)
*wlan = 0;
else
*wlan = result & WLAN_MASK;
}
ret = rfkill_register(wifi_rfkill);
if (ret)
goto err_wifi;

if (bluetooth) {
if ((result & KILLSWITCH_MASK) == 0)
*bluetooth = 0;
else
*bluetooth = (result & BT_MASK) >> 1;
bt_rfkill = rfkill_alloc("compal-bluetooth", &compal_device->dev,
RFKILL_TYPE_BLUETOOTH, &compal_rfkill_ops,
(void *) BT_MASK);
if (!bt_rfkill) {
ret = -ENOMEM;
goto err_allocate_bt;
}
ret = rfkill_register(bt_rfkill);
if (ret)
goto err_register_bt;

return 0;

err_register_bt:
rfkill_destroy(bt_rfkill);

err_allocate_bt:
rfkill_unregister(wifi_rfkill);

err_wifi:
rfkill_destroy(wifi_rfkill);

return ret;
}

/* Backlight device stuff */
Expand All @@ -169,86 +175,6 @@ static struct backlight_ops compalbl_ops = {

static struct backlight_device *compalbl_device;

/* Platform device */

static ssize_t show_wlan(struct device *dev,
struct device_attribute *attr, char *buf)
{
int ret, enabled;

ret = get_wireless_state(&enabled, NULL);
if (ret < 0)
return ret;

return sprintf(buf, "%i\n", enabled);
}

static ssize_t show_raw(struct device *dev,
struct device_attribute *attr, char *buf)
{
u8 result;

ec_read(COMPAL_EC_COMMAND_WIRELESS, &result);

return sprintf(buf, "%i\n", result);
}

static ssize_t show_bluetooth(struct device *dev,
struct device_attribute *attr, char *buf)
{
int ret, enabled;

ret = get_wireless_state(NULL, &enabled);
if (ret < 0)
return ret;

return sprintf(buf, "%i\n", enabled);
}

static ssize_t store_wlan_state(struct device *dev,
struct device_attribute *attr, const char *buf, size_t count)
{
int state, ret;

if (sscanf(buf, "%i", &state) != 1 || (state < 0 || state > 1))
return -EINVAL;

ret = set_wlan_state(state);
if (ret < 0)
return ret;

return count;
}

static ssize_t store_bluetooth_state(struct device *dev,
struct device_attribute *attr, const char *buf, size_t count)
{
int state, ret;

if (sscanf(buf, "%i", &state) != 1 || (state < 0 || state > 1))
return -EINVAL;

ret = set_bluetooth_state(state);
if (ret < 0)
return ret;

return count;
}

static DEVICE_ATTR(bluetooth, 0644, show_bluetooth, store_bluetooth_state);
static DEVICE_ATTR(wlan, 0644, show_wlan, store_wlan_state);
static DEVICE_ATTR(raw, 0444, show_raw, NULL);

static struct attribute *compal_attributes[] = {
&dev_attr_bluetooth.attr,
&dev_attr_wlan.attr,
&dev_attr_raw.attr,
NULL
};

static struct attribute_group compal_attribute_group = {
.attrs = compal_attributes
};

static struct platform_driver compal_driver = {
.driver = {
Expand All @@ -257,8 +183,6 @@ static struct platform_driver compal_driver = {
}
};

static struct platform_device *compal_device;

/* Initialization */

static int dmi_check_cb(const struct dmi_system_id *id)
Expand Down Expand Up @@ -389,23 +313,21 @@ static int __init compal_init(void)

ret = platform_device_add(compal_device);
if (ret)
goto fail_platform_device1;
goto fail_platform_device;

ret = sysfs_create_group(&compal_device->dev.kobj,
&compal_attribute_group);
ret = setup_rfkill();
if (ret)
goto fail_platform_device2;
goto fail_rfkill;

printk(KERN_INFO "compal-laptop: driver "COMPAL_DRIVER_VERSION
" successfully loaded.\n");

return 0;

fail_platform_device2:

fail_rfkill:
platform_device_del(compal_device);

fail_platform_device1:
fail_platform_device:

platform_device_put(compal_device);

Expand All @@ -423,10 +345,13 @@ static int __init compal_init(void)
static void __exit compal_cleanup(void)
{

sysfs_remove_group(&compal_device->dev.kobj, &compal_attribute_group);
platform_device_unregister(compal_device);
platform_driver_unregister(&compal_driver);
backlight_device_unregister(compalbl_device);
rfkill_unregister(wifi_rfkill);
rfkill_destroy(wifi_rfkill);
rfkill_unregister(bt_rfkill);
rfkill_destroy(bt_rfkill);

printk(KERN_INFO "compal-laptop: driver unloaded.\n");
}
Expand Down

0 comments on commit da305af

Please sign in to comment.