[<prev] [next>] [thread-next>] [day] [month] [year] [list]
Message-ID: <200703051532096508636@gmail.com>
Date: Mon, 5 Mar 2007 15:32:15 +0800
From: "Li Yu" <raise.sail@...il.com>
To: "Dmitry Torokhov" <dmitry.torokhov@...il.com>,
"Greg Kroah Hartman" <greg@...ah.com>,
"linux-usb-devel" <linux-usb-devel@...ts.sourceforge.net>,
"LKML" <linux-kernel@...r.kernel.org>,
"Vincent Legoll" <vincentlegoll@...il.com>,
"Zephaniah E. Hull" <warp@...allh.com>,
"Andrew Morton" <akpm@...ux-foundation.org>,
"Harold Sargeant" <harold-sargeant@...world.com>
Cc: "liyu" <liyu@...ss.com.cn>
Subject: [DOC] The documentation for HID Simple Driver Interface 0.5.0
==================================
HID device simple driver interface
==================================
------------------------
Note
------------------------
If you just begin to study from writing input device driver, please see the
input-programming.txt, I am afraid this is not you want, do not confuse with the
"simple" in this name.
------------------------
Version
------------------------
This is used for the version 0.5.0
--------------------------
Overview
--------------------------
Under standard HID device driver development means, we need to write
one interrupt handler for each new HID device to report event to
input subsystem. However, although the most of they can not merge into
mainstream kernel tree, they have only some extended keys, e.g. many
remote controllers. I think it seem break a fly on the wheel, which
write one new interrupt handler for this reason.
The basic idea is reuse the interrupt handler in hid-core.c. so writing
driver for new simple HID device will be more easier, quickly, and do not
need touch hid core.
In essence, this interface just is some hooks in HID core.
The hid-simple.h include this API. But, defore use this interface, you must
include hid.h first.
------------------------
What's you will get from this interface.
------------------------
Use me, you can:
1. Write the driver of USB input device without write code take care of
details of setup and clear usage.
2. A driver use this interface can be as a HID device input filter.
This interface can not support the more drivers handle one device at same time
yet. I can not sure if we really need this feature.
-------------------------
Prepare
-------------------------
Before use this interface, you must turn on these kernel configuration
items:
CONFIG_HID_SIMPLE : HID simple driver interface
--------------------------
Register and unregister
--------------------------
1. Register a simple driver.
int hidinput_register_simple_driver(struct hidinput_simple_driver *simple)
Like any driver register function, it register your driver to kernel,
when the chance that match device come, your driver can probe it. At least, you
must fill the member of parameter simple : name.
Return 0 mean register successfully. elsewise mean there have some
failures in register process.
So far, this function only can return 0 and -EINVAL.
When you driver matched one device, it will reregister it into input
subsystem (see the function hidinput_reconnect_core(), if you want).
This function generally is used in HID sublayer.
2. Unregister a simple driver.
void hidinput_unregister_simple_driver(struct hidinput_simple_driver *simple)
Like any driver register function, it clean your driver from kernel, and
in this process, your driver will disconnect any device which it matched. However,
it do not free your memory of driver structure, that's your task.
This may reregister the device into input subsystem (see the function
hidinput_reconnect_core(), if you want).
This function generally is used in HID sublayer.
3. USB device driver.
You should use follow functions for USB device. the usage of these functions is
same with aboves.
int hidinput_register_simple_usb_driver(struct hidinput_simple_driver *simple)
void hidinput_unregister_simple_usb_driver(struct hidinput_simple_driver *simple)
Both functions is used in driver.
----------------------------------
Usage
----------------------------------
Each simple driver have one data structure hidinput_simple_driver:
struct hidinput_simple_driver {
/*
* The members for implement only are ignored here,
* please do not depend on them
*/
/* public for HID sublayer. */
int (*match_device)(struct hidinput_simple_driver *drv,
struct matched_device *dev);
/* public for drivers */
struct module *owner;
char *name;
int (*connect)(struct hid_device *, struct hid_input *);
void (*disconnect)(struct hid_device *, struct hid_input *);
void (*setup_usage)(struct hid_field *, struct hid_usage *);
void (*clear_usage)(struct hid_field *, struct hid_usage *);
int (*pre_event)(const struct hid_device *, const struct hid_field *,
const struct hid_usage *, const __s32);
int (*post_event)(const struct hid_device *, const struct hid_field *,
const struct hid_usage *, const __s32);
int (*open)(struct input_dev *dev);
void (*close)(struct input_dev *dev);
void *private;
struct usb_device_id *id_table;
struct usage_page_block *usage_page_table;
};
The data member description:
struct module *owner;
In most cases, set this member to THIS_MODULE is your want to.
char *name;
The name of your driver. you must fill this member before register it.
void *private;
You can save the data that your driver use only here. HID simple driver
core do not take care of this.
struct usb_device_id *id_table;
As same with other USB device driver, this is used by matching device,
if so, then call probe()/connect() methods of your driver.
In general, you always want to fill this member with something.
struct usage_page_block *usage_page_table;
Totally, there are three means you can complete setup and clean HID usage work,
which use this member is one of them. Moreover, this means is the most simplest one.
By fill this member, kernel will complete setup and clean usage work automatically.
It will spend many words to explain details, I think the best documentation
of this is the example, you can find them in usbnek4k.c and btp2118.c. I believe
you have enough intelligence to understand them, but if you had familiar
with HID first. :) When you read these examples, you also will find out some other
related data structures ,however, don't worry, they are too simple to not discuss
them.
--------------------------------
Generic Methods
--------------------------------
The simple driver methods description:
Although this simple driver have not direct relation with Linux
device driver architecture, but I still make its API like it on purpose.
The simple driver have follow methods:
1. int (*connect)(struct hid_device *, struct hid_input *);
2. void (*disconnect)(struct hid_device *, struct hid_input *);
When you simple driver is to bind with one real HID device, we will
call connect() method first. To return 0 flag that it complete its mission
successfully, so we can continue, return any other value is looked as
error, any processing do not come.
When the device which your simple driver connect with is down, or
this simple driver is unregistered, we will call disconnect() method first.
3. void (*setup_usage)(struct hid_field *, struct hid_usage *);
4. void (*clear_usage)(struct hid_field *, struct hid_usage *);
The setup_usage() method like hidinput_configure_usage() in
hid_input.c. You also can setup input_dev here. I think
you should be fill the pointer slot for this method, or fill usage_page_table
member of hidinput_simple_driver, elsewise the event() method do not work for
you at all. Please see example in "MS Natural Ergonomic Keyboard 4000" driver.
The clear_usage() method is used to clear side-effect that came from
setup_usage() method when one driver disconnect one device.
Of course, you can do same things in connect/disconnect() method, but
these method can make your life more simpler.
At last, you can use these two methods and usage_page_table member
at same time. These two methods always will be called after kernel process
usage_page_table.
6. int (*pre_event)(const struct hid_device *, const struct hid_field*,
const struct hid_usage *, const __s32);
First, you can use this method send event to input subsystem,
moreover, you can use this as one usage code filter: this method is called
before all other event handling. If it return non-zero , any other event
handling do not be processed, even the hid-input itself. If this method
return zero, the normally event handling process will continue.
See the note [1].
7. void (*post_event)(const struct hid_device *, const struct hid_field *,
const struct hid_usage *, const __s32);
This is called after all other event handling. So if the pre_event() method
return non-zero value, this method also do not called at all.
See the note [1].
8. int (*open)(struct input_dev *dev);
9. void (*close)(struct input_dev *dev);
They are same with corresponding methods of struct input_dev. I assume
you already familiar with them.
10. int (*match_device)(struct hidinput_simple_driver *drv, struct matched_device *dev);
Return non-zero mean drv is matched for dev. An example of its implementation can be
found in drivers/usb/input/hid-core.c
=================================================================================
NOTE:
[1] If you do not configure usage correctly, this method do not work as
you want.
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
Powered by blists - more mailing lists