Drivers Wist Laptops & Desktops



ideapad-slidebar is a new driver which enables slidebars on some
Lenovo IdeaPad laptops.
Fixes this: https://bugs.launchpad.net/ubuntu/+source/linux/+bug/543119
Registers 'IdeaPad Slidebar' input device and
/sys/devices/platform/ideapad_slidebar/slidebar_mode
for switching slidebar's modes.
Now works on:
* Lenovo IdeaPad Y550P
The developing git tree as reference:
git://kernel.ubuntu.com/ikepanhc/ideapad-laptop.git
This patch made against linux-3.10.4.
Git tree for reference: https://github.com/o2genum/ideapad-slidebar.git
Signed-off-by: Andrey Moiseev <o2g....@gmail.com>
---
diff -upNr /usr/src/linux-3.10.4/drivers/input/misc/ideapad_slidebar.c
linux-3.10.4-devel/drivers/input/misc/ideapad_slidebar.c
--- /usr/src/linux-3.10.4/drivers/input/misc/ideapad_slidebar.c
1970-01-01 03:00:00.000000000 +0300
+++ linux-3.10.4-devel/drivers/input/misc/ideapad_slidebar.c
2013-07-30 17:51:32.511678658 +0400
@@ -0,0 +1,354 @@
+/*
+ * Input driver for slidebars on some Lenovo IdeaPad laptops
+ *
+ * Copyright (C) 2013 Andrey Moiseev <o2g....@gmail.com>
+ *
+ * Reverse-engineered from Lenovo SlideNav software (SBarHook.dll).
+ *
+ * This program is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License as published by the Free
+ * Software Foundation; either version 2 of the License, or (at your option)
+ * any later version.
+ *
+ * Trademarks are the property of their respective owners.
+ */
+
+/* Currently tested and works on:
+ * Lenovo IdeaPad Y550P
+ *
+ * Other models can be added easily. To test,
+ * load with 'force' parameter set 'true'.
+ *
+ * LEDs blinking and input mode are managed via sysfs,
+ * (hex, unsigned byte value):
+ * /sys/devices/platform/ideapad_slidebar/slidebar_mode
+ *
+ * The value is in byte range, however, I only figured out
+ * how bits 0b10011001 work. Some other bits, probably,
+ * are meaningfull too.
+ *
+ * Possible states:
+ *
+ * STD_INT, ONMOV_INT, OFF_INT, LAST_POLL, OFF_POLL
+ *
+ * Meaning:
+ * released touched
+ * STD 'heartbeat' lights follow the finger
+ * ONMOV no lights lights follow the finger
+ * LAST at last pos lights follow the finger
+ * OFF no lights no lights
+ *
+ * INT all input events are generated, interrupts are used
+ * POLL no input events by default, to get them, send 0b10000000
(read below)
+ *
+ * Commands: write
+ *
+ * All | 0b01001 -> STD_INT
+ * possible | 0b10001 -> ONMOV_INT
+ * states | 0b01000 -> OFF_INT
+ *
+ * | 0b0 -> LAST_POLL
+ * STD_INT or ONMOV_INT |
+ * | 0b1 -> STD_INT
+ *
+ * | 0b0 -> OFF_POLL
+ * OFF_INT or OFF_POLL |
+ * | 0b1 -> OFF_INT
+ *
+ * Any state | 0b10000000 -> if the slidebar has updated data, produce one
+ * input event (last position), switch to
respective
+ * POLL mode (like 0x0), if not in POLL mode yet.
+ *
+ * Get current state: read
+ *
+ * masked by 0x11 read value means:
+ *
+ * 0x00 LAST
+ * 0x01 STD
+ * 0x10 OFF
+ * 0x11 ONMOV
+ */
+
+#include <linux/module.h>
+#include <linux/kernel.h>
+#include <linux/dmi.h>
+#include <linux/spinlock.h>
+#include <linux/platform_device.h>
+#include <linux/interrupt.h>
+#include <linux/irq.h>
+#include <linux/input.h>
+#include <asm/io.h>
+
+static bool force;
+module_param(force, bool, 0);
+MODULE_PARM_DESC(force, 'Force driver load, ignore DMI data');
+
+#define KBD_IRQ 1
+
+spinlock_t sio_lock = __SPIN_LOCK_UNLOCKED(sio_lock);
+
+static int prev_scancode = 0x00;
+static int touched = 0;
+static struct input_dev *slidebar_input_dev;
+static struct platform_device *slidebar_platform_dev;
+
+/* Hardware interacting */
+static int slidebar_pos_get(void)
+{
+ int res;
+ unsigned long flags;
+ spin_lock_irqsave(&sio_lock, flags);
+ outb(0xf4, 0xff29);
+ outb(0xbf, 0xff2a);
+ res = inb(0xff2b);
+ spin_unlock_irqrestore(&sio_lock, flags);
+ return res;
+}
+
+static unsigned char slidebar_mode_get(void)
+{
+ int res;
+ unsigned long flags;
+ spin_lock_irqsave(&sio_lock, flags);
+ outb(0xf7, 0xff29);
+ outb(0x8b, 0xff2a);
+ res = inb(0xff2b);
+ spin_unlock_irqrestore(&sio_lock, flags);
+ return res;
+}
+
+static void slidebar_mode_set(unsigned char mode)
+{
+ unsigned long flags;
+ spin_lock_irqsave(&sio_lock, flags);
+ outb(0xf7, 0xff29);
+ outb(0x8b, 0xff2a);
+ outb(mode, 0xff2b);
+ spin_unlock_irqrestore(&sio_lock, flags);
+}
+
+/* Keyboard handler */
+static irq_handler_t kbd_irq_handler(int irq, void *dev_id,
+ struct pt_regs *regs)
+{
+ /* Scancodes: e03b on move, bb on release */
+ int scancode = inb(0x60);
+ if (scancode 0xbb) {
+ touched = 0;
+ input_report_key(slidebar_input_dev, BTN_TOUCH, 0);
+ input_sync(slidebar_input_dev);
+ } else if (prev_scancode 0xe0 && scancode 0x3b) {
+ if(!touched) input_report_key(slidebar_input_dev, BTN_TOUCH, 1);
+ touched = 1;
+ input_report_abs(slidebar_input_dev, ABS_X, slidebar_pos_get());
+ input_sync(slidebar_input_dev);
+ }
+ prev_scancode = scancode;
+ return (irq_handler_t) IRQ_HANDLED;
+}
+
+/* Input device */
+static int setup_input_dev(void)
+{
+ int err;
+ err = request_irq(KBD_IRQ, (irq_handler_t) kbd_irq_handler, IRQF_SHARED,
+ 'ideapad_slidebar', (void *)(kbd_irq_handler));
+ if (err) {
+ pr_err('ideapad_slidebar: Can't allocate irq %dn', KBD_IRQ);
+ return -EBUSY;
+ }
+
+ slidebar_input_dev = input_allocate_device();
+ if (!slidebar_input_dev) {
+ pr_err('ideapad_slidebar: Not enough memoryn');
+ err = -ENOMEM;
+ goto err_free_irq;
+ }
+
+ slidebar_input_dev->name = 'IdeaPad Slidebar';
+ slidebar_input_dev->id.bustype = BUS_HOST;
+ slidebar_input_dev->dev.parent = &slidebar_platform_dev->dev;
+ input_set_capability(slidebar_input_dev, EV_KEY, BTN_TOUCH);
+ input_set_capability(slidebar_input_dev, EV_ABS, ABS_X);
+ input_alloc_absinfo(slidebar_input_dev);
+ input_set_abs_params(slidebar_input_dev, ABS_X, 0, 0xff, 0, 0);
+
+ err = input_register_device(slidebar_input_dev);
+ if (err) {
+ pr_err('ideapad_slidebar: Failed to register devicen');
+ goto err_free_dev;
+ }
+ return 0;
+
+ err_free_dev:
+ input_free_device(slidebar_input_dev);
+ err_free_irq:
+ free_irq(KBD_IRQ, (void *)(kbd_irq_handler));
+ return err;
+}
+
+static void remove_input_dev(void)
+{
+ free_irq(KBD_IRQ, (void *)(kbd_irq_handler));
+ input_unregister_device(slidebar_input_dev);
+ input_free_device(slidebar_input_dev);
+}
+
+/* Sysfs slidebar_mode interface */
+static ssize_t show_slidebar_mode(struct device *dev,
+ struct device_attribute *attr,
+ char *buf)
+{
+ return sprintf(buf, '%xn', slidebar_mode_get());
+}
+
+static ssize_t store_slidebar_mode(struct device *dev,
+ struct device_attribute *attr,
+ const char *buf, size_t count) {
+ int mode;
+ if (!count)
+ return 0;
+
+ if(sscanf(buf, '%x', &mode) != 1)
+ return -EINVAL;
+
+ slidebar_mode_set(mode);
+ return count;
+}
+
+static DEVICE_ATTR(slidebar_mode, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH,
+ show_slidebar_mode, store_slidebar_mode);
+
+static struct attribute *ideapad_attributes[] = {
+ &dev_attr_slidebar_mode.attr,
+ NULL
+};
+
+static struct attribute_group ideapad_attribute_group = {
+ .attrs = ideapad_attributes
+};
+
+static int setup_slidebar_mode_dev(void) {
+ return sysfs_create_group(&slidebar_platform_dev->dev.kobj,
+ &ideapad_attribute_group);
+}
+
+static void remove_slidebar_mode_dev(void) {
+ return sysfs_remove_group(&slidebar_platform_dev->dev.kobj,
+ &ideapad_attribute_group);
+}
+
+/* Platform device */
+static int setup_platform_dev(void) {
+ int err;
+ slidebar_platform_dev = platform_device_alloc('ideapad_slidebar', -1);
+ if (!slidebar_platform_dev) {
+ pr_err('ideapad_slidebar: Not enough memoryn');
+ return -ENOMEM;
+ }
+
+ err = platform_device_add(slidebar_platform_dev);
+ if (err) {
+ pr_err('ideapad_slidebar: Failed to register plarform devicen');
+ goto err_free_platform_device;
+ }
+ return 0;
+
+ err_free_platform_device:
+ platform_device_put(slidebar_platform_dev);
+ return err;
+}
+
+static void remove_platform_dev(void) {
+ platform_device_del(slidebar_platform_dev);
+ platform_device_put(slidebar_platform_dev);
+}
+
+/* Platform driver */
+static struct platform_driver slidebar_drv = {
+ .driver = {
+ .name = 'ideapad_slidebar',
+ .owner = THIS_MODULE,
+ },
+};
+
+static int register_platform_drv(void) {
+ int err;
+ err = platform_driver_register(&slidebar_drv);
+ if (err) {
+ pr_err('ideapad_slidebar: Failed to register platform drivern');
+ }
+ return err;
+}
+
+static void unregister_platform_drv(void) {
+ platform_driver_unregister(&slidebar_drv);
+}
+
+/* DMI */
+static int ideapad_dmi_check(const struct dmi_system_id *id) {
+ pr_info('ideapad_slidebar: Laptop model '%s'n', id->ident);
+ return 1;
+}
+
+static struct dmi_system_id __initdata ideapad_dmi_table[] = {
+ {
+ .ident = 'Lenovo IdeaPad Y550P',
+ .matches = {
+ DMI_MATCH(DMI_SYS_VENDOR, 'LENOVO'),
+ DMI_MATCH(DMI_PRODUCT_NAME, '20035'),
+ DMI_MATCH(DMI_PRODUCT_VERSION, 'Lenovo IdeaPad Y550P')
+ },
+ .callback = ideapad_dmi_check
+ }
+};
+
+/* Init and cleanup */
+static int __init slidebar_init(void) {
+ int err;
+
+ if(!force && !dmi_check_system(ideapad_dmi_table))
+ return -ENODEV;
+
+ err = setup_platform_dev();
+ if (err)
+ return err;
+
+ err = register_platform_drv();
+ if (err)
+ goto err_remove_platform_dev;
+
+ err = setup_input_dev();
+ if (err)
+ goto err_unregister_platform_drv;
+
+ err = setup_slidebar_mode_dev();
+ if (err)
+ goto err_remove_input_dev;
+ return 0;
+
+ err_remove_input_dev:
+ remove_input_dev();
+ err_unregister_platform_drv:
+ unregister_platform_drv();
+ err_remove_platform_dev:
+ remove_platform_dev();
+ return err;
+}
+
+static void __exit slidebar_exit(void) {
+ remove_slidebar_mode_dev();
+ remove_input_dev();
+ remove_platform_dev();
+ unregister_platform_drv();
+}
+
+module_init(slidebar_init);
+module_exit(slidebar_exit);
+
+MODULE_AUTHOR('Andrey Moiseev <o2g....@gmail.com>');
+MODULE_DESCRIPTION('Slidebar input support for some Lenovo IdeaPad laptops');
+MODULE_LICENSE('GPL');
+MODULE_VERSION('0.1');
+
+MODULE_ALIAS('dmi:*:svnLENOVO:pn20035:pvrLenovoIdeaPadY550P:*');
diff -upNr /usr/src/linux-3.10.4/drivers/input/misc/Kconfig
linux-3.10.4-devel/drivers/input/misc/Kconfig
--- /usr/src/linux-3.10.4/drivers/input/misc/Kconfig 2013-07-29
03:30:49.000000000 +0400
+++ linux-3.10.4-devel/drivers/input/misc/Kconfig 2013-07-30
17:50:14.215430337 +0400
@@ -637,4 +637,13 @@ config INPUT_XEN_KBDDEV_FRONTEND
To compile this driver as a module, choose M here: the
module will be called xen-kbdfront.
+config INPUT_IDEAPAD_SLIDEBAR
+ tristate 'IdeaPad Laptop Slidebar'
+ depends on INPUT
+ help
+ Input driver for slidebars on some Lenovo IdeaPad laptops.
+
+ If you have an IdeaPad laptop with a slidebar, say Y or M here.
+ Module name is ideapad_slidebar.
+
endif
diff -upNr /usr/src/linux-3.10.4/drivers/input/misc/Makefile
linux-3.10.4-devel/drivers/input/misc/Makefile
--- /usr/src/linux-3.10.4/drivers/input/misc/Makefile 2013-07-29
03:30:49.000000000 +0400
+++ linux-3.10.4-devel/drivers/input/misc/Makefile 2013-07-30
17:50:14.218763647 +0400
@@ -60,3 +60,4 @@ obj-$(CONFIG_INPUT_WISTRON_BTNS) += wist
obj-$(CONFIG_INPUT_WM831X_ON) += wm831x-on.o
obj-$(CONFIG_INPUT_XEN_KBDDEV_FRONTEND) += xen-kbdfront.o
obj-$(CONFIG_INPUT_YEALINK) += yealink.o
+obj-$(CONFIG_INPUT_IDEAPAD_SLIDEBAR) += ideapad_slidebar.o
--
To unsubscribe from this list: send the line 'unsubscribe linux-kernel' in
the body of a message to majo...@vger.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/

Drivers Wist Laptops & Desktops Computers

Laptops

Drivers Wist Laptops & Desktops

  • Products
    All-In-One Products
    Wise Care 365
    Data & Security Products
    Wise Anti Malware
    Wise Data Recovery
    Wise Folder Hider Free
    Wise Folder Hider Pro
    Performance Products
    Wise Registry Cleaner
    Wise Disk Cleaner
    Wise Program Uninstaller
    Wise Memory Optimizer
    Wise Game Booster
    Wise System Monitor
    Wise PC 1stAid
    Wise Plugin Manager
    Utility Tools
    Wise Windows Key Finder
    Wise Duplicate Finder
    Wise Hotkey
    Wise Reminder
    Wise JetSearch
    Wise Auto Shutdown
    Wise Force Deleter
    Multimedia Tools
    Wise YouTube Downloader
    Wise Video Converter
    Wise Video Player
  • Support
    Support Center
    FAQs
    Lost License
    Renew License
    Transfer License
    Feedback
    Product Manuals
    Forgot WFH Login Pasword
    Language
    Languages
    WiseCleaner Forum
    News & Offers
    WiseCleaner Advices & Suggestions
    Wise Care 365
    Wise Registry Cleaner
    Wise Disk Cleaner
    Other WiseCleaner Products
  • Articles
    How-to
    Restore Files with Deep Scan
    Fix Windows 10 Slow Boot-up Issue
    Remove Windows 10 Built-in Apps
    Find and Remove Duplicate Photos
    Protecting Sensitive Files
    Assign Keyboard Shortcut to Launch App
    Set a Reminder in Windows 10
    >> More How-To Articles
    Windows Tips
    Disable Preloading Microsoft Edge at Startup
    High Memory Usage Issue about EoAExperiences.exe
    Restore Deleted Files with Windows File Recovery
    Start Windows 10 in Safe Mode
    God Mode of Windows 10
    Add Windows Photo Viewer in Windows 10
    Prevent PUA with Windows Defender
    >> More Windows Tips
Drivers Wist Laptops & Desktops

Drivers Wist Laptops & Desktops Refurbished

Drivewise ® from Allstate Drive safely and see how much you can save. With Drivewise, you can save for safe driving and receive personalized driving insights. We'll reward you with savings just for activating Drivewise in the Allstate ® mobile app and continue to reward you with cash back every six months for safe driving. Skil tools history of innovation began with the Model E cicular saw in 1924. For over 90 years, Skil has continued to manufacture portable circular saws, as well as drills, impact drivers, grinders and sanders, and more that set industry standards for quality and performance. Drivers free download - CopyTrans Drivers Installer, Adaptec ASPI Drivers, My Drivers, and many more programs.





Comments are closed.