Compare commits
36 Commits
master
...
hack-lenov
Author | SHA1 | Date | |
---|---|---|---|
|
7862f115de | ||
|
251a7fbec4 | ||
|
81e1e7ee4d | ||
|
18c3892778 | ||
|
560eea85ce | ||
|
4feced8a87 | ||
|
8f98977ad8 | ||
|
be80d89a51 | ||
|
1b9e7e6af9 | ||
|
db6eaf14de | ||
|
a5d7192767 | ||
|
bac9a9ac27 | ||
|
a516f95781 | ||
|
d349fc3547 | ||
|
ac2969eea2 | ||
|
94765f1952 | ||
|
5d170444eb | ||
|
2d00d70ec7 | ||
|
f5a4f5f364 | ||
|
aaa0bd3be8 | ||
|
232ee2dc11 | ||
|
82648190aa | ||
|
f4128e5b40 | ||
|
13392b5125 | ||
|
6a1de878c3 | ||
|
e5456d1e0e | ||
|
1e54d10115 | ||
|
6864cf40a4 | ||
|
411db223bb | ||
|
50970d8cf1 | ||
|
ec42e5b7ad | ||
|
a0fdcedf27 | ||
|
205c7ccddc | ||
|
95712ab77f | ||
|
b47ba4b087 | ||
|
3668be9736 |
39
README.md
39
README.md
@ -59,6 +59,45 @@ To uninstall it, run:
|
|||||||
|
|
||||||
# make -f Makefile.dkms uninstall
|
# make -f Makefile.dkms uninstall
|
||||||
|
|
||||||
|
Lenovo IdeaPad Y470/Y570 and Toshiba SATELLITE P870
|
||||||
|
---------------------------------------------------
|
||||||
|
[This kernel bug](https://bugzilla.kernel.org/show_bug.cgi?id=42696) is fixed in
|
||||||
|
Linux 3.9-rc1 and Linux 3.8.5 which obsoletes this hack. Linux 3.7 is
|
||||||
|
incompatible with this hack (and already EOL'd, so a backport fix won't be made
|
||||||
|
available).
|
||||||
|
|
||||||
|
In kernel version 3.6 and older, you need to apply an ugly hack on these laptops
|
||||||
|
to make bbswitch and the
|
||||||
|
driver (both nouveau and nvidia) work. For now I have decided not to put the
|
||||||
|
hack in the bbswitch module since it is a very ugly hack that is comparable to
|
||||||
|
writing a maximum allowable speed of 130 km/h on a traffic sign for a road
|
||||||
|
where 120 km/h is allowed just because the radar gun does not work properly.
|
||||||
|
|
||||||
|
The module has been tested on a Lenovo IdeaPad Y570 running an up-to-date
|
||||||
|
version of Ubuntu 11.10 Oneiric (64-bit) with Bumblebee 3.0 (3.0-1~oneiricppa2)
|
||||||
|
installed using the nvidia driver.
|
||||||
|
|
||||||
|
To make use of it, use the `hack-lenovo` branch. An example using DKMS:
|
||||||
|
|
||||||
|
$ git clone git://github.com/Bumblebee-Project/bbswitch.git -b hack-lenovo
|
||||||
|
$ cd bbswitch
|
||||||
|
$ mkdir /usr/src/acpi-handle-hack-0.0.2
|
||||||
|
# cp Makefile acpi-handle-hack.c /usr/src/acpi-handle-hack-0.0.2
|
||||||
|
# cp dkms/acpi-handle-hack.conf /usr/src/acpi-handle-hack-0.0.2/dkms.conf
|
||||||
|
# dkms install -m acpi-handle-hack -v 0.0.2
|
||||||
|
If everything goes well, you now need to get the hack loaded on boot. On
|
||||||
|
Ubuntu and Debian, this can be done with:
|
||||||
|
|
||||||
|
echo acpi-handle-hack | sudo tee -a /etc/modules
|
||||||
|
sudo update-initramfs -u
|
||||||
|
For other systems, adopt the instructions from the *Disable card on boot*
|
||||||
|
section below. Please do not copy these instructions to blogs/forums/whatever
|
||||||
|
without warning that the method is a hack (you can refer to the metaphore above)
|
||||||
|
and that it may crash the machine if incorrectly applied. To apply these
|
||||||
|
changes, you have to reboot (technically, unloading nvidia/nouveau, bbswitch and
|
||||||
|
stopping bumblebeed, `modprobe acpi-handle-hack` and starting bumblebeed should
|
||||||
|
work as well, but saying reboot is shorter)
|
||||||
|
|
||||||
Usage
|
Usage
|
||||||
-----
|
-----
|
||||||
|
|
||||||
|
134
acpi-handle-hack.c
Normal file
134
acpi-handle-hack.c
Normal file
@ -0,0 +1,134 @@
|
|||||||
|
/**
|
||||||
|
* Very ugly hack to work around a wrongly detected ACPI handle, see
|
||||||
|
* https://bugzilla.kernel.org/show_bug.cgi?id=42696
|
||||||
|
* https://bugzilla.kernel.org/show_bug.cgi?id=60829
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
||||||
|
|
||||||
|
#include <linux/module.h>
|
||||||
|
#include <linux/pci.h>
|
||||||
|
#include <linux/acpi.h>
|
||||||
|
#include <linux/dmi.h>
|
||||||
|
|
||||||
|
MODULE_LICENSE("GPL");
|
||||||
|
MODULE_DESCRIPTION("Dirty ACPI handle hack for Lenovo IdeaPad Y[45]70");
|
||||||
|
MODULE_AUTHOR("Peter Lekensteyn <lekensteyn@gmail.com>");
|
||||||
|
MODULE_VERSION("0.0.2");
|
||||||
|
|
||||||
|
static struct pci_dev *dis_dev;
|
||||||
|
static acpi_handle orig_handle;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns true if the system needs an ACPI handle hack
|
||||||
|
*/
|
||||||
|
static bool __init need_acpi_handle_hack(void) {
|
||||||
|
return dmi_match(DMI_PRODUCT_VERSION, "Lenovo IdeaPad Y470 ")
|
||||||
|
|| dmi_match(DMI_PRODUCT_VERSION, "Lenovo IdeaPad Y480")
|
||||||
|
|| dmi_match(DMI_PRODUCT_VERSION, "Lenovo IdeaPad Y570 ")
|
||||||
|
|| dmi_match(DMI_PRODUCT_VERSION, "LENOVO IDEAPAD Y570 ") /* sys-product-name: PIQY0 */
|
||||||
|
|| dmi_match(DMI_PRODUCT_VERSION, "Lenovo IdeaPad Y580")
|
||||||
|
|| dmi_match(DMI_PRODUCT_VERSION, "Lenovo IdeaPad U510")
|
||||||
|
|| dmi_match(DMI_PRODUCT_VERSION, "PSPLBE-01V00HFR") /* TOSHIBA SATELLITE P870 */
|
||||||
|
|| dmi_match(DMI_PRODUCT_VERSION, "PSPLBA-02300S") /* TOSHIBA Satellite P870 */
|
||||||
|
|| dmi_match(DMI_PRODUCT_VERSION, "PSPLFE-00E009FR") /* TOSHIBA Satellite P870 */
|
||||||
|
|| dmi_match(DMI_PRODUCT_VERSION, "Lenovo G580")
|
||||||
|
|| dmi_match(DMI_PRODUCT_VERSION, "Lenovo G780")
|
||||||
|
|| dmi_match(DMI_PRODUCT_VERSION, "Lenovo IdeaPad Z500")
|
||||||
|
|| (dmi_match(DMI_SYS_VENDOR, "LENOVO") && dmi_match(DMI_PRODUCT_NAME, "PIQY0")) /* Lenovo IdeaPad Y570 */
|
||||||
|
|| dmi_match(DMI_PRODUCT_NAME, "Aspire V5-573G")
|
||||||
|
|| dmi_match(DMI_PRODUCT_NAME, "Aspire V5-573PG")
|
||||||
|
;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct pci_dev __init *get_discrete_device(void) {
|
||||||
|
struct pci_dev *pdev = NULL;
|
||||||
|
while ((pdev = pci_get_device(PCI_ANY_ID, PCI_ANY_ID, pdev)) != NULL) {
|
||||||
|
int pci_class = pdev->class >> 8;
|
||||||
|
|
||||||
|
if (pci_class != PCI_CLASS_DISPLAY_VGA &&
|
||||||
|
pci_class != PCI_CLASS_DISPLAY_3D)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (pdev->vendor != PCI_VENDOR_ID_INTEL) {
|
||||||
|
return pdev;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Very ugly hack to set the ACPI handle, do not use this as exemplary code!
|
||||||
|
*/
|
||||||
|
static void dev_set_acpi_handle(struct pci_dev *pdev, acpi_handle handle) {
|
||||||
|
#ifdef ACPI_HANDLE_SET
|
||||||
|
ACPI_HANDLE_SET(&pdev->dev, handle);
|
||||||
|
#else
|
||||||
|
/* for Linux 3.7 and earlier */
|
||||||
|
pdev->dev.archdata.acpi_handle = handle;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
static int __init hack_apply(void) {
|
||||||
|
acpi_handle tmp_handle, new_handle;
|
||||||
|
struct acpi_buffer buf = { ACPI_ALLOCATE_BUFFER, NULL };
|
||||||
|
if (!need_acpi_handle_hack()) {
|
||||||
|
pr_err("Machine does not need ACPI handle hack\n");
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
dis_dev = get_discrete_device();
|
||||||
|
if (!dis_dev) {
|
||||||
|
pr_err("No discrete video card found\n");
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef ACPI_HANDLE
|
||||||
|
/* since Linux 3.8 */
|
||||||
|
orig_handle = ACPI_HANDLE(&dis_dev->dev);
|
||||||
|
#else
|
||||||
|
/* removed since Linux 3.13 */
|
||||||
|
orig_handle = DEVICE_ACPI_HANDLE(&dis_dev->dev);
|
||||||
|
#endif
|
||||||
|
if (!orig_handle) {
|
||||||
|
pr_err("No ACPI handle found for discrete video card\n");
|
||||||
|
goto free_dev;
|
||||||
|
}
|
||||||
|
if (ACPI_FAILURE(acpi_get_name(orig_handle, ACPI_SINGLE_NAME, &buf))) {
|
||||||
|
pr_err("Could not acquire name for discrete video card\n");
|
||||||
|
goto free_dev;
|
||||||
|
}
|
||||||
|
if (strcmp((char *)buf.pointer, "PEGP") == 0) {
|
||||||
|
pr_err("Handle has already been changed to PEGP\n");
|
||||||
|
goto free_name;
|
||||||
|
}
|
||||||
|
/* \_SB.PCI0.PEG0.VGA_ -> \_SB.PCI0.PEG0.PEGP */
|
||||||
|
if (ACPI_FAILURE(acpi_get_parent(orig_handle, &tmp_handle))) {
|
||||||
|
pr_err("No parent device found for %s\n", (char *)buf.pointer);
|
||||||
|
goto free_name;
|
||||||
|
}
|
||||||
|
if (ACPI_FAILURE(acpi_get_handle(tmp_handle, "PEGP", &new_handle))) {
|
||||||
|
pr_err("No PEGP handle found on %s\n", (char *)buf.pointer);
|
||||||
|
goto free_name;
|
||||||
|
}
|
||||||
|
pr_info("Setting new ACPI handle for discrete video card\n");
|
||||||
|
dev_set_acpi_handle(dis_dev, new_handle);
|
||||||
|
kfree(buf.pointer);
|
||||||
|
pci_dev_put(dis_dev);
|
||||||
|
return 0;
|
||||||
|
free_name:
|
||||||
|
kfree(buf.pointer);
|
||||||
|
free_dev:
|
||||||
|
pci_dev_put(dis_dev);
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void __exit hack_undo(void) {
|
||||||
|
if (orig_handle) {
|
||||||
|
pr_info("Restoring original ACPI handle for discrete"
|
||||||
|
" video card\n");
|
||||||
|
dev_set_acpi_handle(dis_dev, orig_handle);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module_init(hack_apply);
|
||||||
|
module_exit(hack_undo);
|
7
dkms/acpi-handle-hack.conf
Normal file
7
dkms/acpi-handle-hack.conf
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
PACKAGE_NAME="acpi-handle-hack"
|
||||||
|
PACKAGE_VERSION="0.0.2"
|
||||||
|
MAKE[0]="make KVERSION=$kernelver modname=acpi-handle-hack"
|
||||||
|
CLEAN="make clean"
|
||||||
|
BUILT_MODULE_NAME[0]="acpi-handle-hack"
|
||||||
|
DEST_MODULE_LOCATION[0]="/kernel/drivers/acpi"
|
||||||
|
AUTOINSTALL="yes"
|
Loading…
x
Reference in New Issue
Block a user