aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--Documentation/thinkpad-acpi.txt61
-rw-r--r--drivers/misc/thinkpad_acpi.c144
-rw-r--r--drivers/misc/thinkpad_acpi.h4
3 files changed, 186 insertions, 23 deletions
diff --git a/Documentation/thinkpad-acpi.txt b/Documentation/thinkpad-acpi.txt
index ebeed589f6d7..2d4803359a04 100644
--- a/Documentation/thinkpad-acpi.txt
+++ b/Documentation/thinkpad-acpi.txt
@@ -225,15 +225,35 @@ sysfs notes:
225 keys mask, and allows one to modify it. 225 keys mask, and allows one to modify it.
226 226
227 227
228Bluetooth -- /proc/acpi/ibm/bluetooth 228Bluetooth
229------------------------------------- 229---------
230 230
231This feature shows the presence and current state of a Bluetooth 231procfs: /proc/acpi/ibm/bluetooth
232device. If Bluetooth is installed, the following commands can be used: 232sysfs device attribute: bluetooth/enable
233
234This feature shows the presence and current state of a ThinkPad
235Bluetooth device in the internal ThinkPad CDC slot.
236
237Procfs notes:
238
239If Bluetooth is installed, the following commands can be used:
233 240
234 echo enable > /proc/acpi/ibm/bluetooth 241 echo enable > /proc/acpi/ibm/bluetooth
235 echo disable > /proc/acpi/ibm/bluetooth 242 echo disable > /proc/acpi/ibm/bluetooth
236 243
244Sysfs notes:
245
246 If the Bluetooth CDC card is installed, it can be enabled /
247 disabled through the "bluetooth/enable" thinkpad-acpi device
248 attribute, and its current status can also be queried.
249
250 enable:
251 0: disables Bluetooth / Bluetooth is disabled
252 1: enables Bluetooth / Bluetooth is enabled.
253
254 Note: this interface will be probably be superseeded by the
255 generic rfkill class.
256
237Video output control -- /proc/acpi/ibm/video 257Video output control -- /proc/acpi/ibm/video
238-------------------------------------------- 258--------------------------------------------
239 259
@@ -874,23 +894,42 @@ with EINVAL, try to set pwm1_enable to 1 and pwm1 to at least 128 (255
874would be the safest choice, though). 894would be the safest choice, though).
875 895
876 896
877EXPERIMENTAL: WAN -- /proc/acpi/ibm/wan 897EXPERIMENTAL: WAN
878--------------------------------------- 898-----------------
899
900procfs: /proc/acpi/ibm/wan
901sysfs device attribute: wwan/enable
879 902
880This feature is marked EXPERIMENTAL because the implementation 903This feature is marked EXPERIMENTAL because the implementation
881directly accesses hardware registers and may not work as expected. USE 904directly accesses hardware registers and may not work as expected. USE
882WITH CAUTION! To use this feature, you need to supply the 905WITH CAUTION! To use this feature, you need to supply the
883experimental=1 parameter when loading the module. 906experimental=1 parameter when loading the module.
884 907
885This feature shows the presence and current state of a WAN (Sierra 908This feature shows the presence and current state of a W-WAN (Sierra
886Wireless EV-DO) device. If WAN is installed, the following commands can 909Wireless EV-DO) device.
887be used: 910
911It was tested on a Lenovo Thinkpad X60. It should probably work on other
912Thinkpad models which come with this module installed.
913
914Procfs notes:
915
916If the W-WAN card is installed, the following commands can be used:
888 917
889 echo enable > /proc/acpi/ibm/wan 918 echo enable > /proc/acpi/ibm/wan
890 echo disable > /proc/acpi/ibm/wan 919 echo disable > /proc/acpi/ibm/wan
891 920
892It was tested on a Lenovo Thinkpad X60. It should probably work on other 921Sysfs notes:
893Thinkpad models which come with this module installed. 922
923 If the W-WAN card is installed, it can be enabled /
924 disabled through the "wwan/enable" thinkpad-acpi device
925 attribute, and its current status can also be queried.
926
927 enable:
928 0: disables WWAN card / WWAN card is disabled
929 1: enables WWAN card / WWAN card is enabled.
930
931 Note: this interface will be probably be superseeded by the
932 generic rfkill class.
894 933
895Multiple Commands, Module Parameters 934Multiple Commands, Module Parameters
896------------------------------------ 935------------------------------------
diff --git a/drivers/misc/thinkpad_acpi.c b/drivers/misc/thinkpad_acpi.c
index 83a8d984e709..6c36a55cb3d1 100644
--- a/drivers/misc/thinkpad_acpi.c
+++ b/drivers/misc/thinkpad_acpi.c
@@ -1020,8 +1020,54 @@ static struct ibm_struct hotkey_driver_data = {
1020 * Bluetooth subdriver 1020 * Bluetooth subdriver
1021 */ 1021 */
1022 1022
1023/* sysfs bluetooth enable ---------------------------------------------- */
1024static ssize_t bluetooth_enable_show(struct device *dev,
1025 struct device_attribute *attr,
1026 char *buf)
1027{
1028 int status;
1029
1030 status = bluetooth_get_radiosw();
1031 if (status < 0)
1032 return status;
1033
1034 return snprintf(buf, PAGE_SIZE, "%d\n", status ? 1 : 0);
1035}
1036
1037static ssize_t bluetooth_enable_store(struct device *dev,
1038 struct device_attribute *attr,
1039 const char *buf, size_t count)
1040{
1041 unsigned long t;
1042 int res;
1043
1044 if (parse_strtoul(buf, 1, &t))
1045 return -EINVAL;
1046
1047 res = bluetooth_set_radiosw(t);
1048
1049 return (res) ? res : count;
1050}
1051
1052static struct device_attribute dev_attr_bluetooth_enable =
1053 __ATTR(enable, S_IWUSR | S_IRUGO,
1054 bluetooth_enable_show, bluetooth_enable_store);
1055
1056/* --------------------------------------------------------------------- */
1057
1058static struct attribute *bluetooth_attributes[] = {
1059 &dev_attr_bluetooth_enable.attr,
1060 NULL
1061};
1062
1063static const struct attribute_group bluetooth_attr_group = {
1064 .name = TPACPI_BLUETH_SYSFS_GROUP,
1065 .attrs = bluetooth_attributes,
1066};
1067
1023static int __init bluetooth_init(struct ibm_init_struct *iibm) 1068static int __init bluetooth_init(struct ibm_init_struct *iibm)
1024{ 1069{
1070 int res;
1025 int status = 0; 1071 int status = 0;
1026 1072
1027 vdbg_printk(TPACPI_DBG_INIT, "initializing bluetooth subdriver\n"); 1073 vdbg_printk(TPACPI_DBG_INIT, "initializing bluetooth subdriver\n");
@@ -1037,17 +1083,29 @@ static int __init bluetooth_init(struct ibm_init_struct *iibm)
1037 str_supported(tp_features.bluetooth), 1083 str_supported(tp_features.bluetooth),
1038 status); 1084 status);
1039 1085
1040 if (tp_features.bluetooth && 1086 if (tp_features.bluetooth) {
1041 !(status & TP_ACPI_BLUETOOTH_HWPRESENT)) { 1087 if (!(status & TP_ACPI_BLUETOOTH_HWPRESENT)) {
1042 /* no bluetooth hardware present in system */ 1088 /* no bluetooth hardware present in system */
1043 tp_features.bluetooth = 0; 1089 tp_features.bluetooth = 0;
1044 dbg_printk(TPACPI_DBG_INIT, 1090 dbg_printk(TPACPI_DBG_INIT,
1045 "bluetooth hardware not installed\n"); 1091 "bluetooth hardware not installed\n");
1092 } else {
1093 res = sysfs_create_group(&tpacpi_pdev->dev.kobj,
1094 &bluetooth_attr_group);
1095 if (res)
1096 return res;
1097 }
1046 } 1098 }
1047 1099
1048 return (tp_features.bluetooth)? 0 : 1; 1100 return (tp_features.bluetooth)? 0 : 1;
1049} 1101}
1050 1102
1103static void bluetooth_exit(void)
1104{
1105 sysfs_remove_group(&tpacpi_pdev->dev.kobj,
1106 &bluetooth_attr_group);
1107}
1108
1051static int bluetooth_get_radiosw(void) 1109static int bluetooth_get_radiosw(void)
1052{ 1110{
1053 int status; 1111 int status;
@@ -1080,6 +1138,7 @@ static int bluetooth_set_radiosw(int radio_on)
1080 return 0; 1138 return 0;
1081} 1139}
1082 1140
1141/* procfs -------------------------------------------------------------- */
1083static int bluetooth_read(char *p) 1142static int bluetooth_read(char *p)
1084{ 1143{
1085 int len = 0; 1144 int len = 0;
@@ -1119,14 +1178,61 @@ static struct ibm_struct bluetooth_driver_data = {
1119 .name = "bluetooth", 1178 .name = "bluetooth",
1120 .read = bluetooth_read, 1179 .read = bluetooth_read,
1121 .write = bluetooth_write, 1180 .write = bluetooth_write,
1181 .exit = bluetooth_exit,
1122}; 1182};
1123 1183
1124/************************************************************************* 1184/*************************************************************************
1125 * Wan subdriver 1185 * Wan subdriver
1126 */ 1186 */
1127 1187
1188/* sysfs wan enable ---------------------------------------------------- */
1189static ssize_t wan_enable_show(struct device *dev,
1190 struct device_attribute *attr,
1191 char *buf)
1192{
1193 int status;
1194
1195 status = wan_get_radiosw();
1196 if (status < 0)
1197 return status;
1198
1199 return snprintf(buf, PAGE_SIZE, "%d\n", status ? 1 : 0);
1200}
1201
1202static ssize_t wan_enable_store(struct device *dev,
1203 struct device_attribute *attr,
1204 const char *buf, size_t count)
1205{
1206 unsigned long t;
1207 int res;
1208
1209 if (parse_strtoul(buf, 1, &t))
1210 return -EINVAL;
1211
1212 res = wan_set_radiosw(t);
1213
1214 return (res) ? res : count;
1215}
1216
1217static struct device_attribute dev_attr_wan_enable =
1218 __ATTR(enable, S_IWUSR | S_IRUGO,
1219 wan_enable_show, wan_enable_store);
1220
1221/* --------------------------------------------------------------------- */
1222
1223static struct attribute *wan_attributes[] = {
1224 &dev_attr_wan_enable.attr,
1225 NULL
1226};
1227
1228static const struct attribute_group wan_attr_group = {
1229 .name = TPACPI_WAN_SYSFS_GROUP,
1230 .attrs = wan_attributes,
1231};
1232
1128static int __init wan_init(struct ibm_init_struct *iibm) 1233static int __init wan_init(struct ibm_init_struct *iibm)
1129{ 1234{
1235 int res;
1130 int status = 0; 1236 int status = 0;
1131 1237
1132 vdbg_printk(TPACPI_DBG_INIT, "initializing wan subdriver\n"); 1238 vdbg_printk(TPACPI_DBG_INIT, "initializing wan subdriver\n");
@@ -1140,17 +1246,29 @@ static int __init wan_init(struct ibm_init_struct *iibm)
1140 str_supported(tp_features.wan), 1246 str_supported(tp_features.wan),
1141 status); 1247 status);
1142 1248
1143 if (tp_features.wan && 1249 if (tp_features.wan) {
1144 !(status & TP_ACPI_WANCARD_HWPRESENT)) { 1250 if (!(status & TP_ACPI_WANCARD_HWPRESENT)) {
1145 /* no wan hardware present in system */ 1251 /* no wan hardware present in system */
1146 tp_features.wan = 0; 1252 tp_features.wan = 0;
1147 dbg_printk(TPACPI_DBG_INIT, 1253 dbg_printk(TPACPI_DBG_INIT,
1148 "wan hardware not installed\n"); 1254 "wan hardware not installed\n");
1255 } else {
1256 res = sysfs_create_group(&tpacpi_pdev->dev.kobj,
1257 &wan_attr_group);
1258 if (res)
1259 return res;
1260 }
1149 } 1261 }
1150 1262
1151 return (tp_features.wan)? 0 : 1; 1263 return (tp_features.wan)? 0 : 1;
1152} 1264}
1153 1265
1266static void wan_exit(void)
1267{
1268 sysfs_remove_group(&tpacpi_pdev->dev.kobj,
1269 &wan_attr_group);
1270}
1271
1154static int wan_get_radiosw(void) 1272static int wan_get_radiosw(void)
1155{ 1273{
1156 int status; 1274 int status;
@@ -1183,6 +1301,7 @@ static int wan_set_radiosw(int radio_on)
1183 return 0; 1301 return 0;
1184} 1302}
1185 1303
1304/* procfs -------------------------------------------------------------- */
1186static int wan_read(char *p) 1305static int wan_read(char *p)
1187{ 1306{
1188 int len = 0; 1307 int len = 0;
@@ -1222,6 +1341,7 @@ static struct ibm_struct wan_driver_data = {
1222 .name = "wan", 1341 .name = "wan",
1223 .read = wan_read, 1342 .read = wan_read,
1224 .write = wan_write, 1343 .write = wan_write,
1344 .exit = wan_exit,
1225 .flags.experimental = 1, 1345 .flags.experimental = 1,
1226}; 1346};
1227 1347
diff --git a/drivers/misc/thinkpad_acpi.h b/drivers/misc/thinkpad_acpi.h
index 7615adb381c8..a6c285585863 100644
--- a/drivers/misc/thinkpad_acpi.h
+++ b/drivers/misc/thinkpad_acpi.h
@@ -278,6 +278,8 @@ static int beep_write(char *buf);
278 * Bluetooth subdriver 278 * Bluetooth subdriver
279 */ 279 */
280 280
281#define TPACPI_BLUETH_SYSFS_GROUP "bluetooth"
282
281enum { 283enum {
282 /* ACPI GBDC/SBDC bits */ 284 /* ACPI GBDC/SBDC bits */
283 TP_ACPI_BLUETOOTH_HWPRESENT = 0x01, /* Bluetooth hw available */ 285 TP_ACPI_BLUETOOTH_HWPRESENT = 0x01, /* Bluetooth hw available */
@@ -551,6 +553,8 @@ static int volume_write(char *buf);
551 * Wan subdriver 553 * Wan subdriver
552 */ 554 */
553 555
556#define TPACPI_WAN_SYSFS_GROUP "wwan"
557
554enum { 558enum {
555 /* ACPI GWAN/SWAN bits */ 559 /* ACPI GWAN/SWAN bits */
556 TP_ACPI_WANCARD_HWPRESENT = 0x01, /* Wan hw available */ 560 TP_ACPI_WANCARD_HWPRESENT = 0x01, /* Wan hw available */