Initial commit
This commit is contained in:
commit
1dbc135efe
340
LICENSE
Normal file
340
LICENSE
Normal file
@ -0,0 +1,340 @@
|
|||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
Version 2, June 1991
|
||||||
|
|
||||||
|
Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
|
||||||
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
Preamble
|
||||||
|
|
||||||
|
The licenses for most software are designed to take away your
|
||||||
|
freedom to share and change it. By contrast, the GNU General Public
|
||||||
|
License is intended to guarantee your freedom to share and change free
|
||||||
|
software--to make sure the software is free for all its users. This
|
||||||
|
General Public License applies to most of the Free Software
|
||||||
|
Foundation's software and to any other program whose authors commit to
|
||||||
|
using it. (Some other Free Software Foundation software is covered by
|
||||||
|
the GNU Lesser General Public License instead.) You can apply it to
|
||||||
|
your programs, too.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom, not
|
||||||
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
|
have the freedom to distribute copies of free software (and charge for
|
||||||
|
this service if you wish), that you receive source code or can get it
|
||||||
|
if you want it, that you can change the software or use pieces of it
|
||||||
|
in new free programs; and that you know you can do these things.
|
||||||
|
|
||||||
|
To protect your rights, we need to make restrictions that forbid
|
||||||
|
anyone to deny you these rights or to ask you to surrender the rights.
|
||||||
|
These restrictions translate to certain responsibilities for you if you
|
||||||
|
distribute copies of the software, or if you modify it.
|
||||||
|
|
||||||
|
For example, if you distribute copies of such a program, whether
|
||||||
|
gratis or for a fee, you must give the recipients all the rights that
|
||||||
|
you have. You must make sure that they, too, receive or can get the
|
||||||
|
source code. And you must show them these terms so they know their
|
||||||
|
rights.
|
||||||
|
|
||||||
|
We protect your rights with two steps: (1) copyright the software, and
|
||||||
|
(2) offer you this license which gives you legal permission to copy,
|
||||||
|
distribute and/or modify the software.
|
||||||
|
|
||||||
|
Also, for each author's protection and ours, we want to make certain
|
||||||
|
that everyone understands that there is no warranty for this free
|
||||||
|
software. If the software is modified by someone else and passed on, we
|
||||||
|
want its recipients to know that what they have is not the original, so
|
||||||
|
that any problems introduced by others will not reflect on the original
|
||||||
|
authors' reputations.
|
||||||
|
|
||||||
|
Finally, any free program is threatened constantly by software
|
||||||
|
patents. We wish to avoid the danger that redistributors of a free
|
||||||
|
program will individually obtain patent licenses, in effect making the
|
||||||
|
program proprietary. To prevent this, we have made it clear that any
|
||||||
|
patent must be licensed for everyone's free use or not licensed at all.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow.
|
||||||
|
|
||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||||
|
|
||||||
|
0. This License applies to any program or other work which contains
|
||||||
|
a notice placed by the copyright holder saying it may be distributed
|
||||||
|
under the terms of this General Public License. The "Program", below,
|
||||||
|
refers to any such program or work, and a "work based on the Program"
|
||||||
|
means either the Program or any derivative work under copyright law:
|
||||||
|
that is to say, a work containing the Program or a portion of it,
|
||||||
|
either verbatim or with modifications and/or translated into another
|
||||||
|
language. (Hereinafter, translation is included without limitation in
|
||||||
|
the term "modification".) Each licensee is addressed as "you".
|
||||||
|
|
||||||
|
Activities other than copying, distribution and modification are not
|
||||||
|
covered by this License; they are outside its scope. The act of
|
||||||
|
running the Program is not restricted, and the output from the Program
|
||||||
|
is covered only if its contents constitute a work based on the
|
||||||
|
Program (independent of having been made by running the Program).
|
||||||
|
Whether that is true depends on what the Program does.
|
||||||
|
|
||||||
|
1. You may copy and distribute verbatim copies of the Program's
|
||||||
|
source code as you receive it, in any medium, provided that you
|
||||||
|
conspicuously and appropriately publish on each copy an appropriate
|
||||||
|
copyright notice and disclaimer of warranty; keep intact all the
|
||||||
|
notices that refer to this License and to the absence of any warranty;
|
||||||
|
and give any other recipients of the Program a copy of this License
|
||||||
|
along with the Program.
|
||||||
|
|
||||||
|
You may charge a fee for the physical act of transferring a copy, and
|
||||||
|
you may at your option offer warranty protection in exchange for a fee.
|
||||||
|
|
||||||
|
2. You may modify your copy or copies of the Program or any portion
|
||||||
|
of it, thus forming a work based on the Program, and copy and
|
||||||
|
distribute such modifications or work under the terms of Section 1
|
||||||
|
above, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) You must cause the modified files to carry prominent notices
|
||||||
|
stating that you changed the files and the date of any change.
|
||||||
|
|
||||||
|
b) You must cause any work that you distribute or publish, that in
|
||||||
|
whole or in part contains or is derived from the Program or any
|
||||||
|
part thereof, to be licensed as a whole at no charge to all third
|
||||||
|
parties under the terms of this License.
|
||||||
|
|
||||||
|
c) If the modified program normally reads commands interactively
|
||||||
|
when run, you must cause it, when started running for such
|
||||||
|
interactive use in the most ordinary way, to print or display an
|
||||||
|
announcement including an appropriate copyright notice and a
|
||||||
|
notice that there is no warranty (or else, saying that you provide
|
||||||
|
a warranty) and that users may redistribute the program under
|
||||||
|
these conditions, and telling the user how to view a copy of this
|
||||||
|
License. (Exception: if the Program itself is interactive but
|
||||||
|
does not normally print such an announcement, your work based on
|
||||||
|
the Program is not required to print an announcement.)
|
||||||
|
|
||||||
|
These requirements apply to the modified work as a whole. If
|
||||||
|
identifiable sections of that work are not derived from the Program,
|
||||||
|
and can be reasonably considered independent and separate works in
|
||||||
|
themselves, then this License, and its terms, do not apply to those
|
||||||
|
sections when you distribute them as separate works. But when you
|
||||||
|
distribute the same sections as part of a whole which is a work based
|
||||||
|
on the Program, the distribution of the whole must be on the terms of
|
||||||
|
this License, whose permissions for other licensees extend to the
|
||||||
|
entire whole, and thus to each and every part regardless of who wrote it.
|
||||||
|
|
||||||
|
Thus, it is not the intent of this section to claim rights or contest
|
||||||
|
your rights to work written entirely by you; rather, the intent is to
|
||||||
|
exercise the right to control the distribution of derivative or
|
||||||
|
collective works based on the Program.
|
||||||
|
|
||||||
|
In addition, mere aggregation of another work not based on the Program
|
||||||
|
with the Program (or with a work based on the Program) on a volume of
|
||||||
|
a storage or distribution medium does not bring the other work under
|
||||||
|
the scope of this License.
|
||||||
|
|
||||||
|
3. You may copy and distribute the Program (or a work based on it,
|
||||||
|
under Section 2) in object code or executable form under the terms of
|
||||||
|
Sections 1 and 2 above provided that you also do one of the following:
|
||||||
|
|
||||||
|
a) Accompany it with the complete corresponding machine-readable
|
||||||
|
source code, which must be distributed under the terms of Sections
|
||||||
|
1 and 2 above on a medium customarily used for software interchange; or,
|
||||||
|
|
||||||
|
b) Accompany it with a written offer, valid for at least three
|
||||||
|
years, to give any third party, for a charge no more than your
|
||||||
|
cost of physically performing source distribution, a complete
|
||||||
|
machine-readable copy of the corresponding source code, to be
|
||||||
|
distributed under the terms of Sections 1 and 2 above on a medium
|
||||||
|
customarily used for software interchange; or,
|
||||||
|
|
||||||
|
c) Accompany it with the information you received as to the offer
|
||||||
|
to distribute corresponding source code. (This alternative is
|
||||||
|
allowed only for noncommercial distribution and only if you
|
||||||
|
received the program in object code or executable form with such
|
||||||
|
an offer, in accord with Subsection b above.)
|
||||||
|
|
||||||
|
The source code for a work means the preferred form of the work for
|
||||||
|
making modifications to it. For an executable work, complete source
|
||||||
|
code means all the source code for all modules it contains, plus any
|
||||||
|
associated interface definition files, plus the scripts used to
|
||||||
|
control compilation and installation of the executable. However, as a
|
||||||
|
special exception, the source code distributed need not include
|
||||||
|
anything that is normally distributed (in either source or binary
|
||||||
|
form) with the major components (compiler, kernel, and so on) of the
|
||||||
|
operating system on which the executable runs, unless that component
|
||||||
|
itself accompanies the executable.
|
||||||
|
|
||||||
|
If distribution of executable or object code is made by offering
|
||||||
|
access to copy from a designated place, then offering equivalent
|
||||||
|
access to copy the source code from the same place counts as
|
||||||
|
distribution of the source code, even though third parties are not
|
||||||
|
compelled to copy the source along with the object code.
|
||||||
|
|
||||||
|
4. You may not copy, modify, sublicense, or distribute the Program
|
||||||
|
except as expressly provided under this License. Any attempt
|
||||||
|
otherwise to copy, modify, sublicense or distribute the Program is
|
||||||
|
void, and will automatically terminate your rights under this License.
|
||||||
|
However, parties who have received copies, or rights, from you under
|
||||||
|
this License will not have their licenses terminated so long as such
|
||||||
|
parties remain in full compliance.
|
||||||
|
|
||||||
|
5. You are not required to accept this License, since you have not
|
||||||
|
signed it. However, nothing else grants you permission to modify or
|
||||||
|
distribute the Program or its derivative works. These actions are
|
||||||
|
prohibited by law if you do not accept this License. Therefore, by
|
||||||
|
modifying or distributing the Program (or any work based on the
|
||||||
|
Program), you indicate your acceptance of this License to do so, and
|
||||||
|
all its terms and conditions for copying, distributing or modifying
|
||||||
|
the Program or works based on it.
|
||||||
|
|
||||||
|
6. Each time you redistribute the Program (or any work based on the
|
||||||
|
Program), the recipient automatically receives a license from the
|
||||||
|
original licensor to copy, distribute or modify the Program subject to
|
||||||
|
these terms and conditions. You may not impose any further
|
||||||
|
restrictions on the recipients' exercise of the rights granted herein.
|
||||||
|
You are not responsible for enforcing compliance by third parties to
|
||||||
|
this License.
|
||||||
|
|
||||||
|
7. If, as a consequence of a court judgment or allegation of patent
|
||||||
|
infringement or for any other reason (not limited to patent issues),
|
||||||
|
conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot
|
||||||
|
distribute so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you
|
||||||
|
may not distribute the Program at all. For example, if a patent
|
||||||
|
license would not permit royalty-free redistribution of the Program by
|
||||||
|
all those who receive copies directly or indirectly through you, then
|
||||||
|
the only way you could satisfy both it and this License would be to
|
||||||
|
refrain entirely from distribution of the Program.
|
||||||
|
|
||||||
|
If any portion of this section is held invalid or unenforceable under
|
||||||
|
any particular circumstance, the balance of the section is intended to
|
||||||
|
apply and the section as a whole is intended to apply in other
|
||||||
|
circumstances.
|
||||||
|
|
||||||
|
It is not the purpose of this section to induce you to infringe any
|
||||||
|
patents or other property right claims or to contest validity of any
|
||||||
|
such claims; this section has the sole purpose of protecting the
|
||||||
|
integrity of the free software distribution system, which is
|
||||||
|
implemented by public license practices. Many people have made
|
||||||
|
generous contributions to the wide range of software distributed
|
||||||
|
through that system in reliance on consistent application of that
|
||||||
|
system; it is up to the author/donor to decide if he or she is willing
|
||||||
|
to distribute software through any other system and a licensee cannot
|
||||||
|
impose that choice.
|
||||||
|
|
||||||
|
This section is intended to make thoroughly clear what is believed to
|
||||||
|
be a consequence of the rest of this License.
|
||||||
|
|
||||||
|
8. If the distribution and/or use of the Program is restricted in
|
||||||
|
certain countries either by patents or by copyrighted interfaces, the
|
||||||
|
original copyright holder who places the Program under this License
|
||||||
|
may add an explicit geographical distribution limitation excluding
|
||||||
|
those countries, so that distribution is permitted only in or among
|
||||||
|
countries not thus excluded. In such case, this License incorporates
|
||||||
|
the limitation as if written in the body of this License.
|
||||||
|
|
||||||
|
9. The Free Software Foundation may publish revised and/or new versions
|
||||||
|
of the General Public License from time to time. Such new versions will
|
||||||
|
be similar in spirit to the present version, but may differ in detail to
|
||||||
|
address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the Program
|
||||||
|
specifies a version number of this License which applies to it and "any
|
||||||
|
later version", you have the option of following the terms and conditions
|
||||||
|
either of that version or of any later version published by the Free
|
||||||
|
Software Foundation. If the Program does not specify a version number of
|
||||||
|
this License, you may choose any version ever published by the Free Software
|
||||||
|
Foundation.
|
||||||
|
|
||||||
|
10. If you wish to incorporate parts of the Program into other free
|
||||||
|
programs whose distribution conditions are different, write to the author
|
||||||
|
to ask for permission. For software which is copyrighted by the Free
|
||||||
|
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||||
|
make exceptions for this. Our decision will be guided by the two goals
|
||||||
|
of preserving the free status of all derivatives of our free software and
|
||||||
|
of promoting the sharing and reuse of software generally.
|
||||||
|
|
||||||
|
NO WARRANTY
|
||||||
|
|
||||||
|
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
||||||
|
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
||||||
|
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
||||||
|
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
||||||
|
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
|
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
||||||
|
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
||||||
|
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
||||||
|
REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||||
|
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||||
|
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
||||||
|
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
||||||
|
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
||||||
|
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
||||||
|
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
||||||
|
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
||||||
|
POSSIBILITY OF SUCH DAMAGES.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
How to Apply These Terms to Your New Programs
|
||||||
|
|
||||||
|
If you develop a new program, and you want it to be of the greatest
|
||||||
|
possible use to the public, the best way to achieve this is to make it
|
||||||
|
free software which everyone can redistribute and change under these terms.
|
||||||
|
|
||||||
|
To do so, attach the following notices to the program. It is safest
|
||||||
|
to attach them to the start of each source file to most effectively
|
||||||
|
convey the exclusion of warranty; and each file should have at least
|
||||||
|
the "copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
|
<one line to give the program's name and a brief idea of what it does.>
|
||||||
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License along
|
||||||
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
|
||||||
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
If the program is interactive, make it output a short notice like this
|
||||||
|
when it starts in an interactive mode:
|
||||||
|
|
||||||
|
Gnomovision version 69, Copyright (C) year name of author
|
||||||
|
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||||
|
This is free software, and you are welcome to redistribute it
|
||||||
|
under certain conditions; type `show c' for details.
|
||||||
|
|
||||||
|
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||||
|
parts of the General Public License. Of course, the commands you use may
|
||||||
|
be called something other than `show w' and `show c'; they could even be
|
||||||
|
mouse-clicks or menu items--whatever suits your program.
|
||||||
|
|
||||||
|
You should also get your employer (if you work as a programmer) or your
|
||||||
|
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||||
|
necessary. Here is a sample; alter the names:
|
||||||
|
|
||||||
|
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||||
|
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||||
|
|
||||||
|
<signature of Ty Coon>, 1 April 1989
|
||||||
|
Ty Coon, President of Vice
|
||||||
|
|
||||||
|
This General Public License does not permit incorporating your program into
|
||||||
|
proprietary programs. If your program is a subroutine library, you may
|
||||||
|
consider it more useful to permit linking proprietary applications with the
|
||||||
|
library. If this is what you want to do, use the GNU Lesser General
|
||||||
|
Public License instead of this License.
|
||||||
|
|
56
Makefile
Normal file
56
Makefile
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
# Uncomment the following to enable debug.
|
||||||
|
#DEBUG = y
|
||||||
|
|
||||||
|
KVER := $(shell uname -r)
|
||||||
|
KSRC := /lib/modules/$(KVER)/build
|
||||||
|
MODDESTDIR := /lib/modules/$(KVER)/kernel/drivers/input/keyboard
|
||||||
|
MODULE_NAME := perixxkbd
|
||||||
|
MODULE_VER := 1.0.0
|
||||||
|
|
||||||
|
ifeq ($(DEBUG),y)
|
||||||
|
DBGFLAGS = -O -g -DML_DEBUG
|
||||||
|
else
|
||||||
|
DBGFLAGS = -O2
|
||||||
|
endif
|
||||||
|
|
||||||
|
ccflags-y += $(DBGFLAGS)
|
||||||
|
|
||||||
|
|
||||||
|
ifneq ($(KERNELRELEASE),)
|
||||||
|
obj-m := $(MODULE_NAME).o
|
||||||
|
else
|
||||||
|
KSRC := /lib/modules/$(KVER)/build
|
||||||
|
PWD := $(shell pwd)
|
||||||
|
endif
|
||||||
|
#@if [ -n $(dkms status $(MODULE_NAME)/$(MODULE_VER)) ]; then \
|
||||||
|
|
||||||
|
define REMOVE_MODULE
|
||||||
|
@if [ -n "`dkms status $(MODULE_NAME)/$(MODULE_VER)`" ]; then \
|
||||||
|
dkms remove $(MODULE_NAME)/$(MODULE_VER) --all; \
|
||||||
|
fi;
|
||||||
|
endef
|
||||||
|
|
||||||
|
default:
|
||||||
|
$(MAKE) -C $(KSRC) M=$(PWD) modules
|
||||||
|
|
||||||
|
clean:
|
||||||
|
$(MAKE) -C $(KSRC) M=$(PWD) clean
|
||||||
|
|
||||||
|
uninstall:
|
||||||
|
rm -f $(MODDESTDIR)/$(MODULE_NAME).ko
|
||||||
|
/sbin/depmod -a ${KVER}
|
||||||
|
|
||||||
|
install:
|
||||||
|
install -p -m 644 $(MODULE_NAME).ko $(MODDESTDIR)
|
||||||
|
/sbin/depmod -a ${KVER}
|
||||||
|
|
||||||
|
dkms: clean
|
||||||
|
rm -rf /usr/src/$(MODULE_NAME)-$(MODULE_VER)
|
||||||
|
mkdir /usr/src/$(MODULE_NAME)-$(MODULE_VER) -p
|
||||||
|
cp . /usr/src/$(MODULE_NAME)-$(MODULE_VER) -a
|
||||||
|
rm -rf /usr/src/$(MODULE_NAME)-$(MODULE_VER)/.hg
|
||||||
|
$(REMOVE_MODULE)
|
||||||
|
dkms add -m $(MODULE_NAME) -v $(MODULE_VER)
|
||||||
|
dkms build -m $(MODULE_NAME) -v $(MODULE_VER)
|
||||||
|
dkms install -m $(MODULE_NAME) -v $(MODULE_VER) --force
|
||||||
|
|
74
README.md
Normal file
74
README.md
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
# Linux Microdia Keyboard Chipset Driver #
|
||||||
|
|
||||||
|
For Chipset `0x0c45`:`0x7603`
|
||||||
|
The kernel reports the chipset as `Microdia`
|
||||||
|
|
||||||
|
Written for the Perixx PX-1800 USB Keyboard: [Perixx PX-1800 Keyboard](http://www.perixx.com/en/service/Perixx_Manual/GAMING/PX-1800.pdf)
|
||||||
|
Original base: swoogan.blogspot.de/2014/09/azio-l70-keyboard-linux-driver.html
|
||||||
|
|
||||||
|
> NOTE: Makefile and instructions are only tested on Ubuntu, however they are known to work on Debian, Arch, Fedora, and Manjaro.
|
||||||
|
|
||||||
|
Reportedy supports the following keyboards as well:
|
||||||
|
|
||||||
|
* SL-6432-BK - Speedlink LUCIDIS Comfort Illuminated Keyboard
|
||||||
|
* COUGAR 200K Scissor Gaming Keyboard
|
||||||
|
* GAMDIAS USB Keyboard (unspecified model but will report as Microdia chipset)
|
||||||
|
* Avazz USB Keyboard (unspecified model but will report as Microdia chipset)
|
||||||
|
* Perixx P1800
|
||||||
|
* Modecom MC800-Volcano
|
||||||
|
|
||||||
|
# Installation ##
|
||||||
|
## DKMS ##
|
||||||
|
|
||||||
|
# debian-based:
|
||||||
|
sudo apt-get install mercurial build-essential linux-headers-generic dkms
|
||||||
|
|
||||||
|
# fedora:
|
||||||
|
sudo dnf install kernel-devel kernel-headers
|
||||||
|
sudo dnf groupinstall "Development Tools" "Development Libraries"
|
||||||
|
|
||||||
|
hg clone https://bitbucket.org/Swoogan/aziokbd
|
||||||
|
cd aziokbd
|
||||||
|
sudo ./install.sh dkms
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## Manual Install ##
|
||||||
|
|
||||||
|
sudo apt-get install mercurial build-essential linux-headers-generic
|
||||||
|
hg clone https://bitbucket.org/Swoogan/aziokbd
|
||||||
|
cd aziokbd
|
||||||
|
sudo ./install.sh
|
||||||
|
|
||||||
|
# Blacklisting #
|
||||||
|
|
||||||
|
**NOTE: install.sh attempts to blacklist the driver for you. You shouldn't need to do anything manually. These instructions are to explain the process, in the event something goes wrong.**
|
||||||
|
|
||||||
|
You need to blacklist the device from the generic USB hid driver in order for the aziokbd driver to control it.
|
||||||
|
|
||||||
|
## Kernel Module ##
|
||||||
|
If the USB hid driver is compiled as a kernel module you will need to create a quirks file and blacklist it there.
|
||||||
|
|
||||||
|
You can determine if the driver is a module by running the following:
|
||||||
|
|
||||||
|
lsmod | grep usbhid
|
||||||
|
|
||||||
|
If `grep` finds something, it means that the driver is a module.
|
||||||
|
|
||||||
|
Create a file called `/etc/modprobe.d/usbhid.conf` and add the following to it:
|
||||||
|
|
||||||
|
options usbhid quirks=0x0c45:0x7603:0x0004
|
||||||
|
|
||||||
|
If you find that the generic USB driver is still taking the device, try changing the `0x0004` to a `0x0007`.
|
||||||
|
|
||||||
|
## Compiled into Kernel ##
|
||||||
|
If the generic USB hid driver is compiled into the kernel, then the driver is not loaded as a module and setting the option via `modprobe` will not work. In this case you must pass the option to the driver via the grub boot loader.
|
||||||
|
|
||||||
|
Create a new file in `/etc/default/grub.d/`. For example, you might call it `aziokbd.conf`. (If your grub package doesn't have this directory, just modify the generic `/etc/default/grub` configuration file):
|
||||||
|
|
||||||
|
GRUB_CMDLINE_LINUX_DEFAULT='usbhid.quirks=0x0c45:0x7603:0x4'
|
||||||
|
|
||||||
|
Then run `sudo update-grub` and reboot.
|
||||||
|
|
||||||
|
Again, if you find that `0x4` doesn't work, try `0x7`.
|
||||||
|
|
6
dkms.conf
Normal file
6
dkms.conf
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
PACKAGE_NAME=perixxkbd
|
||||||
|
PACKAGE_VERSION=1.0.0
|
||||||
|
BUILT_MODULE_NAME[0]=perixxkbd
|
||||||
|
DEST_MODULE_LOCATION[0]="/kernel/drivers/input/keyboard"
|
||||||
|
AUTOINSTALL="yes"
|
||||||
|
|
62
install.sh
Executable file
62
install.sh
Executable file
@ -0,0 +1,62 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
if [[ $1 != 'dkms' ]]; then
|
||||||
|
echo '## Making package ##'
|
||||||
|
make
|
||||||
|
|
||||||
|
echo '## Installing package ##'
|
||||||
|
make install
|
||||||
|
else
|
||||||
|
echo '## Installing package with DKMS ##'
|
||||||
|
make dkms
|
||||||
|
fi
|
||||||
|
|
||||||
|
quirk='0x0c45:0x7603:0x0007'
|
||||||
|
modquirk="options usbhid quirks=$quirk"
|
||||||
|
grubquirk="usbhid.quirks=$quirk"
|
||||||
|
|
||||||
|
if (lsmod | grep 'usbhid'); then
|
||||||
|
echo '## usbhid is module ##'
|
||||||
|
|
||||||
|
# Making sure the quirk does not get added multiple times
|
||||||
|
if ! (cat /etc/modprobe.d/usbhid.conf | grep "$modquirk"); then
|
||||||
|
echo '## Writing to /etc/modprobe.d/usbhid.conf ##'
|
||||||
|
echo $modquirk >> /etc/modprobe.d/usbhid.conf
|
||||||
|
if [[ $1 != 'dkms' ]]; then sudo echo 'perixxkbd' >> /etc/modules; fi
|
||||||
|
else
|
||||||
|
echo 'NOTICE - modprobe config files have already been updated'
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo '## Starting module ##'
|
||||||
|
modprobe perixxkbd
|
||||||
|
|
||||||
|
# Note: this line may fail if you have other drivers loaded that depend
|
||||||
|
# on usbhid. For example, your mouse driver. In that case you would have
|
||||||
|
# to remove those drivers first, then load them again.
|
||||||
|
|
||||||
|
echo '## Attempting to reload usbhid module ##'
|
||||||
|
rmmod usbhid && modprobe usbhid quirks=$quirk
|
||||||
|
else
|
||||||
|
echo '## usbhid is compiled into kernel ##'
|
||||||
|
|
||||||
|
# Making sure the quirk does not get added multiple times
|
||||||
|
if ! (cat /etc/default/grub.d/perixxkbd.conf | grep "$grubquirk"); then
|
||||||
|
echo '## Writing to /etc/default/grub.d/perixxkbd.conf ##'
|
||||||
|
echo $grubquirk >> /etc/default/grub.d/perixxkbd.conf
|
||||||
|
$distro = $(lsb_release -si)
|
||||||
|
if ($distro | grep 'Ubuntu'); then
|
||||||
|
update-grub
|
||||||
|
fi
|
||||||
|
if [ -f "/etc/arch-release" ]; then
|
||||||
|
update-grub
|
||||||
|
fi
|
||||||
|
if [ -f "/etc/fedora-release" ]; then
|
||||||
|
grub2-mkconfig -o /boot/grub2/grub.cfg
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
echo 'NOTICE - grub config file has already been updated'
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo '## You must reboot to load the module ##'
|
||||||
|
fi
|
||||||
|
|
494
perixxkbd.c
Normal file
494
perixxkbd.c
Normal file
@ -0,0 +1,494 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2013 Colin Svingen
|
||||||
|
* Copyright (c) 2016 Christoph Haas
|
||||||
|
*
|
||||||
|
* Perixx PX-1800 USB Keyboard support
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*
|
||||||
|
* Should you need to contact me, the author, you can do so by email.
|
||||||
|
* Mail your message to Christoph Haas <christoph.h@sprinternet.at>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
||||||
|
|
||||||
|
#include <linux/kernel.h>
|
||||||
|
#include <linux/slab.h>
|
||||||
|
#include <linux/module.h>
|
||||||
|
#include <linux/init.h>
|
||||||
|
#include <linux/usb/input.h>
|
||||||
|
#include <linux/hid.h>
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Version Information
|
||||||
|
*/
|
||||||
|
#define DRIVER_VERSION ""
|
||||||
|
#define DRIVER_AUTHOR "Christoph Haas <christoph.h@sprinternet.at>"
|
||||||
|
#define DRIVER_DESC "Perixx PX-1800 Keyboard Driver"
|
||||||
|
#define DRIVER_LICENSE "GPL"
|
||||||
|
#define ML_VENDOR_ID 0x0c45
|
||||||
|
#define ML_PRODUCT_ID 0x7603
|
||||||
|
|
||||||
|
MODULE_AUTHOR(DRIVER_AUTHOR);
|
||||||
|
MODULE_DESCRIPTION(DRIVER_DESC);
|
||||||
|
MODULE_LICENSE(DRIVER_LICENSE);
|
||||||
|
|
||||||
|
static const unsigned char px_kbd_keycode[256] = {
|
||||||
|
/* BEGIN 04 */
|
||||||
|
/* 0-7 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 8-15 */ KEY_LEFTCTRL, KEY_LEFTSHIFT, KEY_LEFTALT, KEY_LEFTMETA, KEY_RIGHTCTRL, KEY_RIGHTSHIFT, KEY_RIGHTALT, KEY_RESERVED,
|
||||||
|
/* 16-23 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_A, KEY_B, KEY_C, KEY_D,
|
||||||
|
/* 24-31 */ KEY_E, KEY_F, KEY_G, KEY_H, KEY_I, KEY_J, KEY_K, KEY_L,
|
||||||
|
/* 32-39 */ KEY_M, KEY_N, KEY_O, KEY_P, KEY_Q, KEY_R, KEY_S, KEY_T,
|
||||||
|
/* 40-47 */ KEY_U, KEY_V, KEY_W, KEY_X, KEY_Y, KEY_Z, KEY_1, KEY_2,
|
||||||
|
/* 48-55 */ KEY_3, KEY_4, KEY_5, KEY_6, KEY_7, KEY_8, KEY_9, KEY_0,
|
||||||
|
/* 56-63 */ KEY_ENTER, KEY_ESC, KEY_BACKSPACE, KEY_TAB, KEY_SPACE, KEY_MINUS, KEY_EQUAL, KEY_LEFTBRACE,
|
||||||
|
/* END 04 */
|
||||||
|
|
||||||
|
/* BEGIN 05 */
|
||||||
|
/* 64-71 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 72-79 */ KEY_RIGHTBRACE, KEY_BACKSLASH, KEY_BACKSLASH, KEY_SEMICOLON, KEY_APOSTROPHE, KEY_GRAVE, KEY_COMMA, KEY_DOT,
|
||||||
|
/* 80-87 */ KEY_SLASH, KEY_CAPSLOCK, KEY_F1, KEY_F2, KEY_F3, KEY_F4, KEY_F5, KEY_F6,
|
||||||
|
/* 88-95 */ KEY_F7, KEY_F8, KEY_F9, KEY_F10, KEY_F11, KEY_F12, KEY_SYSRQ, KEY_SCROLLLOCK,
|
||||||
|
/* 96-103 */ KEY_PAUSE, KEY_INSERT, KEY_HOME, KEY_PAGEUP, KEY_DELETE, KEY_END, KEY_PAGEDOWN, KEY_RIGHT,
|
||||||
|
/* 104-111 */ KEY_LEFT, KEY_DOWN, KEY_UP, KEY_NUMLOCK, KEY_KPSLASH, KEY_KPASTERISK, KEY_KPMINUS, KEY_KPPLUS,
|
||||||
|
/* 112-119 */ KEY_KPENTER, KEY_KP1, KEY_KP2, KEY_KP3, KEY_KP4, KEY_KP5, KEY_KP6, KEY_KP7,
|
||||||
|
/* 120-127 */ KEY_KP8, KEY_KP9, KEY_KP0, KEY_KPDOT, KEY_102ND, KEY_MENU, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* END 05 */
|
||||||
|
|
||||||
|
/* 01 */
|
||||||
|
/* 128-135 */ KEY_VOLUMEDOWN, KEY_VOLUMEUP, KEY_MEDIA, KEY_MUTE, KEY_PAUSE, KEY_PREVIOUSSONG, KEY_PLAYPAUSE, KEY_NEXTSONG,
|
||||||
|
/* 136-143 */ KEY_MAIL, KEY_HOMEPAGE, KEY_CALC, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 144-151 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 152-159 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 160-167 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 168-175 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 176-183 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 184-191 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
|
||||||
|
/* BEGIN 06 */
|
||||||
|
/* 192-199 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 200-207 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 208-215 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 216-223 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 224-231 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_SLASH,
|
||||||
|
/* 232-239 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 240-247 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* 248-255 */ KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED, KEY_RESERVED,
|
||||||
|
/* END 06 */
|
||||||
|
};
|
||||||
|
|
||||||
|
struct usb_kbd {
|
||||||
|
struct input_dev *dev;
|
||||||
|
struct usb_device *usbdev;
|
||||||
|
unsigned char old[8];
|
||||||
|
struct urb *irq, *led;
|
||||||
|
unsigned char newleds;
|
||||||
|
char name[128];
|
||||||
|
char phys[64];
|
||||||
|
|
||||||
|
unsigned char *new;
|
||||||
|
struct usb_ctrlrequest *cr;
|
||||||
|
unsigned char *leds;
|
||||||
|
dma_addr_t new_dma;
|
||||||
|
dma_addr_t leds_dma;
|
||||||
|
};
|
||||||
|
|
||||||
|
static void usb_kbd_irq(struct urb *urb)
|
||||||
|
{
|
||||||
|
struct usb_kbd *kbd = urb->context;
|
||||||
|
int i, j, offset;
|
||||||
|
char keys[9];
|
||||||
|
|
||||||
|
switch (urb->status) {
|
||||||
|
case 0: /* success */
|
||||||
|
break;
|
||||||
|
case -ECONNRESET: /* unlink */
|
||||||
|
case -ENOENT:
|
||||||
|
case -ESHUTDOWN:
|
||||||
|
return;
|
||||||
|
/* -EPIPE: should clear the halt */
|
||||||
|
default: /* error */
|
||||||
|
goto resubmit;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
printk("Keyup keycode: ");
|
||||||
|
|
||||||
|
for (i = 0; i < 8; i++)
|
||||||
|
printk("%d ", kbd->old[i]);
|
||||||
|
|
||||||
|
printk("\n");
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
// The following lines are for logging keypresses to the
|
||||||
|
// kernel dmesg facility. Uncomment the following lines
|
||||||
|
// to capture the keycode for any non-functioning keys
|
||||||
|
// and open a new issue on bitbucket.org with the key
|
||||||
|
// you pressed and the keycode output below.
|
||||||
|
|
||||||
|
printk("Keydown keycode: ");
|
||||||
|
|
||||||
|
for (i = 0; i < 8; i++)
|
||||||
|
printk("%d ", kbd->new[i]);
|
||||||
|
|
||||||
|
printk("\n");
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
if (kbd->new[0] == 1) {
|
||||||
|
// volume down
|
||||||
|
if (kbd->new[1] == 234 && kbd->old[1] != 234)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[128], 1);
|
||||||
|
if (kbd->old[1] == 234 && kbd->new[1] != 234)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[128], 0);
|
||||||
|
|
||||||
|
// volume up
|
||||||
|
if (kbd->new[1] == 233 && kbd->old[1] != 233)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[129], 1);
|
||||||
|
if (kbd->old[1] == 233 && kbd->new[1] != 233)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[129], 0);
|
||||||
|
|
||||||
|
// Media
|
||||||
|
if (kbd->new[1] == 131 && kbd->old[1] != 131)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[130], 1);
|
||||||
|
if (kbd->old[1] == 131 && kbd->new[1] != 131)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[130], 0);
|
||||||
|
|
||||||
|
// Mute
|
||||||
|
if (kbd->new[1] == 226 && kbd->old[1] != 226)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[131], 1);
|
||||||
|
if (kbd->old[1] == 226 && kbd->new[1] != 226)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[131], 0);
|
||||||
|
|
||||||
|
// Stop
|
||||||
|
if (kbd->new[1] == 183 && kbd->old[1] != 183)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[132], 1);
|
||||||
|
if (kbd->old[1] == 183 && kbd->new[1] != 183)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[132], 0);
|
||||||
|
|
||||||
|
// Prev Song
|
||||||
|
if (kbd->new[1] == 182 && kbd->old[1] != 182)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[133], 1);
|
||||||
|
if (kbd->old[1] == 182 && kbd->new[1] != 182)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[133], 0);
|
||||||
|
|
||||||
|
// Play/Pause
|
||||||
|
if (kbd->new[1] == 205 && kbd->old[1] != 205)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[134], 1);
|
||||||
|
if (kbd->old[1] == 205 && kbd->new[1] != 205)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[134], 0);
|
||||||
|
|
||||||
|
// Next Song
|
||||||
|
if (kbd->new[1] == 181 && kbd->old[1] != 181)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[135], 1);
|
||||||
|
if (kbd->old[1] == 181 && kbd->new[1] != 181)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[135], 0);
|
||||||
|
|
||||||
|
// Mail
|
||||||
|
if (kbd->new[1] == 138 && kbd->old[1] != 138)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[136], 1);
|
||||||
|
if (kbd->old[1] == 138 && kbd->new[1] != 138)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[136], 0);
|
||||||
|
|
||||||
|
// Homepage
|
||||||
|
if (kbd->new[1] == 35 && kbd->old[1] != 35)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[137], 1);
|
||||||
|
if (kbd->old[1] == 35 && kbd->new[1] != 35)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[137], 0);
|
||||||
|
|
||||||
|
// Calc
|
||||||
|
if (kbd->new[1] == 146 && kbd->old[1] != 146)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[138], 1);
|
||||||
|
if (kbd->old[1] == 146 && kbd->new[1] != 146)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[138], 0);
|
||||||
|
}
|
||||||
|
else if (kbd->new[0] == 4) {
|
||||||
|
for (j = 1; j < 8; j++) {
|
||||||
|
offset = j * 8;
|
||||||
|
for (i = 0; i < 8; i++)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[offset + i], (kbd->new[j] >> i) & 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (kbd->new[0] == 5) {
|
||||||
|
for (j = 1; j < 8; j++) {
|
||||||
|
offset = (j * 8) + 64;
|
||||||
|
for (i = 0; i < 8; i++)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[offset + i], (kbd->new[j] >> i) & 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (kbd->new[0] == 6) {
|
||||||
|
for (j = 1; j < 8; j++) {
|
||||||
|
offset = (j * 8) + 192;
|
||||||
|
for (i = 0; i < 8; i++)
|
||||||
|
input_report_key(kbd->dev, px_kbd_keycode[offset + i], (kbd->new[j] >> i) & 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
input_sync(kbd->dev);
|
||||||
|
|
||||||
|
memcpy(kbd->old, kbd->new, 8);
|
||||||
|
|
||||||
|
resubmit:
|
||||||
|
i = usb_submit_urb (urb, GFP_ATOMIC);
|
||||||
|
if (i)
|
||||||
|
hid_err(urb->dev, "can't resubmit intr, %s-%s/input0, status %d",
|
||||||
|
kbd->usbdev->bus->bus_name,
|
||||||
|
kbd->usbdev->devpath, i);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int usb_kbd_event(struct input_dev *dev, unsigned int type,
|
||||||
|
unsigned int code, int value)
|
||||||
|
{
|
||||||
|
struct usb_kbd *kbd = input_get_drvdata(dev);
|
||||||
|
|
||||||
|
if (type != EV_LED)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
kbd->newleds = (!!test_bit(LED_KANA, dev->led) << 3) | (!!test_bit(LED_COMPOSE, dev->led) << 3) |
|
||||||
|
(!!test_bit(LED_SCROLLL, dev->led) << 2) | (!!test_bit(LED_CAPSL, dev->led) << 1) |
|
||||||
|
(!!test_bit(LED_NUML, dev->led));
|
||||||
|
|
||||||
|
if (kbd->led->status == -EINPROGRESS)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (*(kbd->leds) == kbd->newleds)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
*(kbd->leds) = kbd->newleds;
|
||||||
|
kbd->led->dev = kbd->usbdev;
|
||||||
|
if (usb_submit_urb(kbd->led, GFP_ATOMIC))
|
||||||
|
pr_err("usb_submit_urb(leds) failed\n");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void usb_kbd_led(struct urb *urb)
|
||||||
|
{
|
||||||
|
struct usb_kbd *kbd = urb->context;
|
||||||
|
|
||||||
|
if (urb->status)
|
||||||
|
hid_warn(urb->dev, "led urb status %d received\n",
|
||||||
|
urb->status);
|
||||||
|
|
||||||
|
if (*(kbd->leds) == kbd->newleds)
|
||||||
|
return;
|
||||||
|
|
||||||
|
*(kbd->leds) = kbd->newleds;
|
||||||
|
kbd->led->dev = kbd->usbdev;
|
||||||
|
|
||||||
|
if (usb_submit_urb(kbd->led, GFP_ATOMIC))
|
||||||
|
hid_err(urb->dev, "usb_submit_urb(leds) failed\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int usb_kbd_open(struct input_dev *dev)
|
||||||
|
{
|
||||||
|
struct usb_kbd *kbd = input_get_drvdata(dev);
|
||||||
|
|
||||||
|
kbd->irq->dev = kbd->usbdev;
|
||||||
|
if (usb_submit_urb(kbd->irq, GFP_KERNEL))
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void usb_kbd_close(struct input_dev *dev)
|
||||||
|
{
|
||||||
|
struct usb_kbd *kbd = input_get_drvdata(dev);
|
||||||
|
|
||||||
|
usb_kill_urb(kbd->irq);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int usb_kbd_alloc_mem(struct usb_device *dev, struct usb_kbd *kbd)
|
||||||
|
{
|
||||||
|
if (!(kbd->irq = usb_alloc_urb(0, GFP_KERNEL)))
|
||||||
|
return -1;
|
||||||
|
if (!(kbd->led = usb_alloc_urb(0, GFP_KERNEL)))
|
||||||
|
return -1;
|
||||||
|
if (!(kbd->new = usb_alloc_coherent(dev, 8, GFP_ATOMIC, &kbd->new_dma)))
|
||||||
|
return -1;
|
||||||
|
if (!(kbd->cr = kmalloc(sizeof(struct usb_ctrlrequest), GFP_KERNEL)))
|
||||||
|
return -1;
|
||||||
|
if (!(kbd->leds = usb_alloc_coherent(dev, 1, GFP_ATOMIC, &kbd->leds_dma)))
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void usb_kbd_free_mem(struct usb_device *dev, struct usb_kbd *kbd)
|
||||||
|
{
|
||||||
|
usb_free_urb(kbd->irq);
|
||||||
|
usb_free_urb(kbd->led);
|
||||||
|
usb_free_coherent(dev, 8, kbd->new, kbd->new_dma);
|
||||||
|
kfree(kbd->cr);
|
||||||
|
usb_free_coherent(dev, 1, kbd->leds, kbd->leds_dma);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int usb_kbd_probe(struct usb_interface *iface,
|
||||||
|
const struct usb_device_id *id)
|
||||||
|
{
|
||||||
|
struct usb_device *dev = interface_to_usbdev(iface);
|
||||||
|
struct usb_host_interface *interface;
|
||||||
|
struct usb_endpoint_descriptor *endpoint;
|
||||||
|
struct usb_kbd *kbd;
|
||||||
|
struct input_dev *input_dev;
|
||||||
|
int i, pipe, maxp;
|
||||||
|
int error = -ENOMEM;
|
||||||
|
|
||||||
|
interface = iface->cur_altsetting;
|
||||||
|
|
||||||
|
if (interface->desc.bNumEndpoints != 1)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
endpoint = &interface->endpoint[0].desc;
|
||||||
|
if (!usb_endpoint_is_int_in(endpoint))
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
pipe = usb_rcvintpipe(dev, endpoint->bEndpointAddress);
|
||||||
|
maxp = usb_maxpacket(dev, pipe, usb_pipeout(pipe));
|
||||||
|
kbd = kzalloc(sizeof(struct usb_kbd), GFP_KERNEL);
|
||||||
|
input_dev = input_allocate_device();
|
||||||
|
if (!kbd || !input_dev)
|
||||||
|
goto fail1;
|
||||||
|
|
||||||
|
if (usb_kbd_alloc_mem(dev, kbd))
|
||||||
|
goto fail2;
|
||||||
|
|
||||||
|
kbd->usbdev = dev;
|
||||||
|
kbd->dev = input_dev;
|
||||||
|
|
||||||
|
if (dev->manufacturer)
|
||||||
|
strlcpy(kbd->name, dev->manufacturer, sizeof(kbd->name));
|
||||||
|
|
||||||
|
if (dev->product) {
|
||||||
|
if (dev->manufacturer)
|
||||||
|
strlcat(kbd->name, " ", sizeof(kbd->name));
|
||||||
|
strlcat(kbd->name, dev->product, sizeof(kbd->name));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!strlen(kbd->name))
|
||||||
|
snprintf(kbd->name, sizeof(kbd->name),
|
||||||
|
"USB HIDBP Keyboard %04x:%04x",
|
||||||
|
le16_to_cpu(dev->descriptor.idVendor),
|
||||||
|
le16_to_cpu(dev->descriptor.idProduct));
|
||||||
|
|
||||||
|
printk("<1>aziokbd: detected %s\n", kbd->name);
|
||||||
|
|
||||||
|
usb_make_path(dev, kbd->phys, sizeof(kbd->phys));
|
||||||
|
strlcat(kbd->phys, "/input0", sizeof(kbd->phys));
|
||||||
|
|
||||||
|
input_dev->name = kbd->name;
|
||||||
|
input_dev->phys = kbd->phys;
|
||||||
|
usb_to_input_id(dev, &input_dev->id);
|
||||||
|
input_dev->dev.parent = &iface->dev;
|
||||||
|
|
||||||
|
input_set_drvdata(input_dev, kbd);
|
||||||
|
|
||||||
|
input_dev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_LED) |
|
||||||
|
BIT_MASK(EV_REP);
|
||||||
|
input_dev->ledbit[0] = BIT_MASK(LED_NUML) | BIT_MASK(LED_CAPSL) |
|
||||||
|
BIT_MASK(LED_SCROLLL) | BIT_MASK(LED_COMPOSE) |
|
||||||
|
BIT_MASK(LED_KANA);
|
||||||
|
|
||||||
|
for (i = 0; i < 255; i++)
|
||||||
|
set_bit(px_kbd_keycode[i], input_dev->keybit);
|
||||||
|
|
||||||
|
clear_bit(0, input_dev->keybit);
|
||||||
|
|
||||||
|
input_dev->event = usb_kbd_event;
|
||||||
|
input_dev->open = usb_kbd_open;
|
||||||
|
input_dev->close = usb_kbd_close;
|
||||||
|
|
||||||
|
usb_fill_int_urb(kbd->irq, dev, pipe,
|
||||||
|
kbd->new, (maxp > 8 ? 8 : maxp),
|
||||||
|
usb_kbd_irq, kbd, endpoint->bInterval);
|
||||||
|
kbd->irq->transfer_dma = kbd->new_dma;
|
||||||
|
kbd->irq->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
|
||||||
|
|
||||||
|
kbd->cr->bRequestType = USB_TYPE_CLASS | USB_RECIP_INTERFACE;
|
||||||
|
kbd->cr->bRequest = 0x09;
|
||||||
|
kbd->cr->wValue = cpu_to_le16(0x200);
|
||||||
|
//kbd->cr->wIndex = cpu_to_le16(interface->desc.bInterfaceNumber);
|
||||||
|
kbd->cr->wIndex = cpu_to_le16(0);
|
||||||
|
kbd->cr->wLength = cpu_to_le16(1);
|
||||||
|
|
||||||
|
usb_fill_control_urb(kbd->led, dev, usb_sndctrlpipe(dev, 0),
|
||||||
|
(void *) kbd->cr, kbd->leds, 1,
|
||||||
|
usb_kbd_led, kbd);
|
||||||
|
kbd->led->transfer_dma = kbd->leds_dma;
|
||||||
|
kbd->led->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
|
||||||
|
|
||||||
|
error = input_register_device(kbd->dev);
|
||||||
|
if (error)
|
||||||
|
goto fail2;
|
||||||
|
|
||||||
|
usb_set_intfdata(iface, kbd);
|
||||||
|
device_set_wakeup_enable(&dev->dev, 1);
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
fail2:
|
||||||
|
usb_kbd_free_mem(dev, kbd);
|
||||||
|
fail1:
|
||||||
|
input_free_device(input_dev);
|
||||||
|
kfree(kbd);
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void usb_kbd_disconnect(struct usb_interface *intf)
|
||||||
|
{
|
||||||
|
struct usb_kbd *kbd = usb_get_intfdata (intf);
|
||||||
|
|
||||||
|
usb_set_intfdata(intf, NULL);
|
||||||
|
if (kbd) {
|
||||||
|
usb_kill_urb(kbd->irq);
|
||||||
|
input_unregister_device(kbd->dev);
|
||||||
|
usb_kbd_free_mem(interface_to_usbdev(intf), kbd);
|
||||||
|
kfree(kbd);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct usb_device_id usb_kbd_id_table [] = {
|
||||||
|
{ USB_DEVICE(ML_VENDOR_ID, ML_PRODUCT_ID) },
|
||||||
|
{ } /* Terminating entry */
|
||||||
|
};
|
||||||
|
|
||||||
|
MODULE_DEVICE_TABLE (usb, usb_kbd_id_table);
|
||||||
|
|
||||||
|
static struct usb_driver usb_kbd_driver = {
|
||||||
|
.name = "perixxkbd",
|
||||||
|
.probe = usb_kbd_probe,
|
||||||
|
.disconnect = usb_kbd_disconnect,
|
||||||
|
.id_table = usb_kbd_id_table,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int __init usb_kbd_init(void)
|
||||||
|
{
|
||||||
|
int result = usb_register(&usb_kbd_driver);
|
||||||
|
if (result == 0)
|
||||||
|
printk(KERN_INFO KBUILD_MODNAME ": " DRIVER_VERSION ":"
|
||||||
|
DRIVER_DESC "\n");
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void __exit usb_kbd_exit(void)
|
||||||
|
{
|
||||||
|
usb_deregister(&usb_kbd_driver);
|
||||||
|
}
|
||||||
|
|
||||||
|
module_init(usb_kbd_init);
|
||||||
|
module_exit(usb_kbd_exit);
|
Loading…
Reference in New Issue
Block a user