162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Support for rfkill on some Fujitsu-Siemens Amilo laptops. 462306a36Sopenharmony_ci * Copyright 2011 Ben Hutchings. 562306a36Sopenharmony_ci * 662306a36Sopenharmony_ci * Based in part on the fsam7440 driver, which is: 762306a36Sopenharmony_ci * Copyright 2005 Alejandro Vidal Mata & Javier Vidal Mata. 862306a36Sopenharmony_ci * and on the fsaa1655g driver, which is: 962306a36Sopenharmony_ci * Copyright 2006 Martin Večeřa. 1062306a36Sopenharmony_ci */ 1162306a36Sopenharmony_ci 1262306a36Sopenharmony_ci#include <linux/module.h> 1362306a36Sopenharmony_ci#include <linux/dmi.h> 1462306a36Sopenharmony_ci#include <linux/i8042.h> 1562306a36Sopenharmony_ci#include <linux/io.h> 1662306a36Sopenharmony_ci#include <linux/moduleparam.h> 1762306a36Sopenharmony_ci#include <linux/platform_device.h> 1862306a36Sopenharmony_ci#include <linux/rfkill.h> 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_ci/* 2162306a36Sopenharmony_ci * These values were obtained from disassembling and debugging the 2262306a36Sopenharmony_ci * PM.exe program installed in the Fujitsu-Siemens AMILO A1655G 2362306a36Sopenharmony_ci */ 2462306a36Sopenharmony_ci#define A1655_WIFI_COMMAND 0x10C5 2562306a36Sopenharmony_ci#define A1655_WIFI_ON 0x25 2662306a36Sopenharmony_ci#define A1655_WIFI_OFF 0x45 2762306a36Sopenharmony_ci 2862306a36Sopenharmony_cistatic int amilo_a1655_rfkill_set_block(void *data, bool blocked) 2962306a36Sopenharmony_ci{ 3062306a36Sopenharmony_ci u8 param = blocked ? A1655_WIFI_OFF : A1655_WIFI_ON; 3162306a36Sopenharmony_ci int rc; 3262306a36Sopenharmony_ci 3362306a36Sopenharmony_ci i8042_lock_chip(); 3462306a36Sopenharmony_ci rc = i8042_command(¶m, A1655_WIFI_COMMAND); 3562306a36Sopenharmony_ci i8042_unlock_chip(); 3662306a36Sopenharmony_ci return rc; 3762306a36Sopenharmony_ci} 3862306a36Sopenharmony_ci 3962306a36Sopenharmony_cistatic const struct rfkill_ops amilo_a1655_rfkill_ops = { 4062306a36Sopenharmony_ci .set_block = amilo_a1655_rfkill_set_block 4162306a36Sopenharmony_ci}; 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_ci/* 4462306a36Sopenharmony_ci * These values were obtained from disassembling the PM.exe program 4562306a36Sopenharmony_ci * installed in the Fujitsu-Siemens AMILO M 7440 4662306a36Sopenharmony_ci */ 4762306a36Sopenharmony_ci#define M7440_PORT1 0x118f 4862306a36Sopenharmony_ci#define M7440_PORT2 0x118e 4962306a36Sopenharmony_ci#define M7440_RADIO_ON1 0x12 5062306a36Sopenharmony_ci#define M7440_RADIO_ON2 0x80 5162306a36Sopenharmony_ci#define M7440_RADIO_OFF1 0x10 5262306a36Sopenharmony_ci#define M7440_RADIO_OFF2 0x00 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_cistatic int amilo_m7440_rfkill_set_block(void *data, bool blocked) 5562306a36Sopenharmony_ci{ 5662306a36Sopenharmony_ci u8 val1 = blocked ? M7440_RADIO_OFF1 : M7440_RADIO_ON1; 5762306a36Sopenharmony_ci u8 val2 = blocked ? M7440_RADIO_OFF2 : M7440_RADIO_ON2; 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_ci outb(val1, M7440_PORT1); 6062306a36Sopenharmony_ci outb(val2, M7440_PORT2); 6162306a36Sopenharmony_ci 6262306a36Sopenharmony_ci /* Check whether the state has changed correctly */ 6362306a36Sopenharmony_ci if (inb(M7440_PORT1) != val1 || inb(M7440_PORT2) != val2) 6462306a36Sopenharmony_ci return -EIO; 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_ci return 0; 6762306a36Sopenharmony_ci} 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_cistatic const struct rfkill_ops amilo_m7440_rfkill_ops = { 7062306a36Sopenharmony_ci .set_block = amilo_m7440_rfkill_set_block 7162306a36Sopenharmony_ci}; 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_cistatic const struct dmi_system_id amilo_rfkill_id_table[] = { 7462306a36Sopenharmony_ci { 7562306a36Sopenharmony_ci .matches = { 7662306a36Sopenharmony_ci DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU SIEMENS"), 7762306a36Sopenharmony_ci DMI_MATCH(DMI_BOARD_NAME, "AMILO A1655"), 7862306a36Sopenharmony_ci }, 7962306a36Sopenharmony_ci .driver_data = (void *)&amilo_a1655_rfkill_ops 8062306a36Sopenharmony_ci }, 8162306a36Sopenharmony_ci { 8262306a36Sopenharmony_ci .matches = { 8362306a36Sopenharmony_ci DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU SIEMENS"), 8462306a36Sopenharmony_ci DMI_MATCH(DMI_BOARD_NAME, "AMILO L1310"), 8562306a36Sopenharmony_ci }, 8662306a36Sopenharmony_ci .driver_data = (void *)&amilo_a1655_rfkill_ops 8762306a36Sopenharmony_ci }, 8862306a36Sopenharmony_ci { 8962306a36Sopenharmony_ci .matches = { 9062306a36Sopenharmony_ci DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU SIEMENS"), 9162306a36Sopenharmony_ci DMI_MATCH(DMI_BOARD_NAME, "AMILO M7440"), 9262306a36Sopenharmony_ci }, 9362306a36Sopenharmony_ci .driver_data = (void *)&amilo_m7440_rfkill_ops 9462306a36Sopenharmony_ci }, 9562306a36Sopenharmony_ci {} 9662306a36Sopenharmony_ci}; 9762306a36Sopenharmony_ci 9862306a36Sopenharmony_cistatic struct platform_device *amilo_rfkill_pdev; 9962306a36Sopenharmony_cistatic struct rfkill *amilo_rfkill_dev; 10062306a36Sopenharmony_ci 10162306a36Sopenharmony_cistatic int amilo_rfkill_probe(struct platform_device *device) 10262306a36Sopenharmony_ci{ 10362306a36Sopenharmony_ci int rc; 10462306a36Sopenharmony_ci const struct dmi_system_id *system_id = 10562306a36Sopenharmony_ci dmi_first_match(amilo_rfkill_id_table); 10662306a36Sopenharmony_ci 10762306a36Sopenharmony_ci if (!system_id) 10862306a36Sopenharmony_ci return -ENXIO; 10962306a36Sopenharmony_ci 11062306a36Sopenharmony_ci amilo_rfkill_dev = rfkill_alloc(KBUILD_MODNAME, &device->dev, 11162306a36Sopenharmony_ci RFKILL_TYPE_WLAN, 11262306a36Sopenharmony_ci system_id->driver_data, NULL); 11362306a36Sopenharmony_ci if (!amilo_rfkill_dev) 11462306a36Sopenharmony_ci return -ENOMEM; 11562306a36Sopenharmony_ci 11662306a36Sopenharmony_ci rc = rfkill_register(amilo_rfkill_dev); 11762306a36Sopenharmony_ci if (rc) 11862306a36Sopenharmony_ci goto fail; 11962306a36Sopenharmony_ci 12062306a36Sopenharmony_ci return 0; 12162306a36Sopenharmony_ci 12262306a36Sopenharmony_cifail: 12362306a36Sopenharmony_ci rfkill_destroy(amilo_rfkill_dev); 12462306a36Sopenharmony_ci return rc; 12562306a36Sopenharmony_ci} 12662306a36Sopenharmony_ci 12762306a36Sopenharmony_cistatic void amilo_rfkill_remove(struct platform_device *device) 12862306a36Sopenharmony_ci{ 12962306a36Sopenharmony_ci rfkill_unregister(amilo_rfkill_dev); 13062306a36Sopenharmony_ci rfkill_destroy(amilo_rfkill_dev); 13162306a36Sopenharmony_ci} 13262306a36Sopenharmony_ci 13362306a36Sopenharmony_cistatic struct platform_driver amilo_rfkill_driver = { 13462306a36Sopenharmony_ci .driver = { 13562306a36Sopenharmony_ci .name = KBUILD_MODNAME, 13662306a36Sopenharmony_ci }, 13762306a36Sopenharmony_ci .probe = amilo_rfkill_probe, 13862306a36Sopenharmony_ci .remove_new = amilo_rfkill_remove, 13962306a36Sopenharmony_ci}; 14062306a36Sopenharmony_ci 14162306a36Sopenharmony_cistatic int __init amilo_rfkill_init(void) 14262306a36Sopenharmony_ci{ 14362306a36Sopenharmony_ci int rc; 14462306a36Sopenharmony_ci 14562306a36Sopenharmony_ci if (dmi_first_match(amilo_rfkill_id_table) == NULL) 14662306a36Sopenharmony_ci return -ENODEV; 14762306a36Sopenharmony_ci 14862306a36Sopenharmony_ci rc = platform_driver_register(&amilo_rfkill_driver); 14962306a36Sopenharmony_ci if (rc) 15062306a36Sopenharmony_ci return rc; 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ci amilo_rfkill_pdev = platform_device_register_simple(KBUILD_MODNAME, 15362306a36Sopenharmony_ci PLATFORM_DEVID_NONE, 15462306a36Sopenharmony_ci NULL, 0); 15562306a36Sopenharmony_ci if (IS_ERR(amilo_rfkill_pdev)) { 15662306a36Sopenharmony_ci rc = PTR_ERR(amilo_rfkill_pdev); 15762306a36Sopenharmony_ci goto fail; 15862306a36Sopenharmony_ci } 15962306a36Sopenharmony_ci 16062306a36Sopenharmony_ci return 0; 16162306a36Sopenharmony_ci 16262306a36Sopenharmony_cifail: 16362306a36Sopenharmony_ci platform_driver_unregister(&amilo_rfkill_driver); 16462306a36Sopenharmony_ci return rc; 16562306a36Sopenharmony_ci} 16662306a36Sopenharmony_ci 16762306a36Sopenharmony_cistatic void __exit amilo_rfkill_exit(void) 16862306a36Sopenharmony_ci{ 16962306a36Sopenharmony_ci platform_device_unregister(amilo_rfkill_pdev); 17062306a36Sopenharmony_ci platform_driver_unregister(&amilo_rfkill_driver); 17162306a36Sopenharmony_ci} 17262306a36Sopenharmony_ci 17362306a36Sopenharmony_ciMODULE_AUTHOR("Ben Hutchings <ben@decadent.org.uk>"); 17462306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 17562306a36Sopenharmony_ciMODULE_DEVICE_TABLE(dmi, amilo_rfkill_id_table); 17662306a36Sopenharmony_ci 17762306a36Sopenharmony_cimodule_init(amilo_rfkill_init); 17862306a36Sopenharmony_cimodule_exit(amilo_rfkill_exit); 179