rebased on the mouse device addon, cleaned, untested
still lacks better logging, thread worker, settings handling, keymap management and that's all! git-svn-id: file:///srv/svn/repos/haiku/trunk/current@8878 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
a4e05e618a
commit
e73537f459
@ -26,9 +26,21 @@
|
||||
/*****************************************************************************/
|
||||
#include "KeyboardInputDevice.h"
|
||||
|
||||
#include <Directory.h>
|
||||
#include <Entry.h>
|
||||
#include <NodeMonitor.h>
|
||||
#include <Path.h>
|
||||
#include <String.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#define DEBUG 1
|
||||
#if DEBUG
|
||||
#define LOG(text) fputs(text, sLogFile); fflush(sLogFile)
|
||||
#else
|
||||
#define LOG(text)
|
||||
#endif
|
||||
|
||||
const static uint32 kSetLeds = 0x2711;
|
||||
const static uint32 kSetRepeatingKey = 0x2712;
|
||||
const static uint32 kSetNonRepeatingKey = 0x2713;
|
||||
@ -36,6 +48,23 @@ const static uint32 kSetKeyRepeatRate = 0x2714;
|
||||
const static uint32 kSetKeyRepeatDelay = 0x2716;
|
||||
const static uint32 kGetNextKey = 0x270f;
|
||||
|
||||
const static uint32 kKeyboardThreadPriority = B_FIRST_REAL_TIME_PRIORITY + 4;
|
||||
const static char *kKeyboardDevicesDirectory = "/dev/input/keyboard";
|
||||
|
||||
// "/dev/" is automatically prepended by StartMonitoringDevice()
|
||||
const static char *kKeyboardDevicesDirectoryUSB = "input/keyboard/usb";
|
||||
|
||||
FILE *KeyboardInputDevice::sLogFile = NULL;
|
||||
|
||||
struct keyboard_device {
|
||||
KeyboardInputDevice *owner;
|
||||
input_device_ref device_ref;
|
||||
char path[B_PATH_NAME_LENGTH];
|
||||
int fd;
|
||||
thread_id device_watcher;
|
||||
kb_settings settings;
|
||||
bool active;
|
||||
};
|
||||
|
||||
extern "C"
|
||||
BInputServerDevice *
|
||||
@ -46,34 +75,29 @@ instantiate_input_device()
|
||||
|
||||
|
||||
KeyboardInputDevice::KeyboardInputDevice()
|
||||
: fThread(-1),
|
||||
fQuit(false)
|
||||
{
|
||||
fFd = open("dev/input/keyboard/ps2/0", O_RDWR);
|
||||
if (fFd >= 0)
|
||||
fThread = spawn_thread(DeviceWatcher, "keyboard watcher thread",
|
||||
B_NORMAL_PRIORITY, this);
|
||||
|
||||
fLogFile = fopen("/boot/home/device_log.log", "w");
|
||||
#if DEBUG
|
||||
if (sLogFile == NULL)
|
||||
sLogFile = fopen("/var/log/keyboard_device_log.log", "a");
|
||||
#endif
|
||||
|
||||
StartMonitoringDevice(kKeyboardDevicesDirectoryUSB);
|
||||
}
|
||||
|
||||
|
||||
KeyboardInputDevice::~KeyboardInputDevice()
|
||||
{
|
||||
if (fThread >= 0) {
|
||||
status_t dummy;
|
||||
wait_for_thread(fThread, &dummy);
|
||||
}
|
||||
StopMonitoringDevice(kKeyboardDevicesDirectoryUSB);
|
||||
|
||||
if (fFd >= 0)
|
||||
close(fFd);
|
||||
|
||||
fclose(fLogFile);
|
||||
#if DEBUG
|
||||
fclose(sLogFile);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
status_t
|
||||
KeyboardInputDevice::InitFromSettings(uint32 opcode)
|
||||
KeyboardInputDevice::InitFromSettings(void *cookie, uint32 opcode)
|
||||
{
|
||||
// retrieve current values
|
||||
|
||||
@ -84,38 +108,45 @@ KeyboardInputDevice::InitFromSettings(uint32 opcode)
|
||||
status_t
|
||||
KeyboardInputDevice::InitCheck()
|
||||
{
|
||||
InitFromSettings();
|
||||
|
||||
input_device_ref keyboard1 = { "Keyboard 1", B_KEYBOARD_DEVICE, (void *)this };
|
||||
|
||||
input_device_ref *devices[2] = { &keyboard1, NULL };
|
||||
|
||||
if (fFd >= 0 && fThread >= 0) {
|
||||
RegisterDevices(devices);
|
||||
return BInputServerDevice::InitCheck();
|
||||
}
|
||||
return B_ERROR;
|
||||
}
|
||||
|
||||
|
||||
status_t
|
||||
KeyboardInputDevice::Start(const char *name, void *cookie)
|
||||
{
|
||||
fputs("Start(", fLogFile);
|
||||
fputs(name, fLogFile);
|
||||
fputs(")\n", fLogFile);
|
||||
resume_thread(fThread);
|
||||
RecursiveScan(kKeyboardDevicesDirectory);
|
||||
|
||||
return B_OK;
|
||||
}
|
||||
|
||||
|
||||
status_t
|
||||
KeyboardInputDevice::Stop(const char *device, void *cookie)
|
||||
KeyboardInputDevice::Start(const char *name, void *cookie)
|
||||
{
|
||||
fputs("Stop()\n", fLogFile);
|
||||
keyboard_device *device = (keyboard_device *)cookie;
|
||||
|
||||
suspend_thread(fThread);
|
||||
char threadName[B_OS_NAME_LENGTH];
|
||||
snprintf(threadName, B_OS_NAME_LENGTH, "%s watcher", name);
|
||||
|
||||
device->active = true;
|
||||
device->device_watcher = spawn_thread(DeviceWatcher, threadName,
|
||||
kKeyboardThreadPriority, device);
|
||||
|
||||
resume_thread(device->device_watcher);
|
||||
|
||||
return B_OK;
|
||||
}
|
||||
|
||||
|
||||
status_t
|
||||
KeyboardInputDevice::Stop(const char *name, void *cookie)
|
||||
{
|
||||
keyboard_device *device = (keyboard_device *)cookie;
|
||||
|
||||
char log[128];
|
||||
snprintf(log, 128, "Stop(%s)\n", name);
|
||||
|
||||
LOG(log);
|
||||
|
||||
device->active = false;
|
||||
if (device->device_watcher >= 0) {
|
||||
status_t dummy;
|
||||
wait_for_thread(device->device_watcher, &dummy);
|
||||
}
|
||||
|
||||
return B_OK;
|
||||
}
|
||||
@ -125,13 +156,16 @@ status_t
|
||||
KeyboardInputDevice::Control(const char *name, void *cookie,
|
||||
uint32 command, BMessage *message)
|
||||
{
|
||||
fputs("Control()\n", fLogFile);
|
||||
char log[128];
|
||||
snprintf(log, 128, "Control(%s, code: %lu)\n", name, command);
|
||||
|
||||
LOG(log);
|
||||
|
||||
if (command == B_NODE_MONITOR)
|
||||
HandleMonitor(message);
|
||||
else if (command >= B_KEY_MAP_CHANGED
|
||||
&& command <= B_KEY_REPEAT_RATE_CHANGED) {
|
||||
InitFromSettings(command);
|
||||
InitFromSettings(cookie, command);
|
||||
}
|
||||
return B_OK;
|
||||
}
|
||||
@ -140,16 +174,135 @@ KeyboardInputDevice::Control(const char *name, void *cookie,
|
||||
status_t
|
||||
KeyboardInputDevice::HandleMonitor(BMessage *message)
|
||||
{
|
||||
int32 opcode = 0;
|
||||
status_t status;
|
||||
if ((status = message->FindInt32("opcode", &opcode)) < B_OK)
|
||||
return status;
|
||||
|
||||
if ((opcode != B_ENTRY_CREATED)
|
||||
&& (opcode != B_ENTRY_REMOVED))
|
||||
return B_OK;
|
||||
|
||||
|
||||
BEntry entry;
|
||||
BPath path;
|
||||
dev_t device;
|
||||
ino_t directory;
|
||||
const char *name = NULL;
|
||||
|
||||
message->FindInt32("device", &device);
|
||||
message->FindInt64("directory", &directory);
|
||||
message->FindString("name", &name);
|
||||
|
||||
entry_ref ref(device, directory, name);
|
||||
|
||||
if ((status = entry.SetTo(&ref)) != B_OK)
|
||||
return status;
|
||||
if ((status = entry.GetPath(&path)) != B_OK)
|
||||
return status;
|
||||
if ((status = path.InitCheck()) != B_OK)
|
||||
return status;
|
||||
|
||||
if (opcode == B_ENTRY_CREATED)
|
||||
AddDevice(path.Path());
|
||||
else
|
||||
RemoveDevice(path.Path());
|
||||
|
||||
return status;
|
||||
}
|
||||
|
||||
|
||||
status_t
|
||||
KeyboardInputDevice::AddDevice(const char *path)
|
||||
{
|
||||
keyboard_device *device = new keyboard_device();
|
||||
if (!device)
|
||||
return B_NO_MEMORY;
|
||||
|
||||
if ((device->fd = open(path, O_RDWR)) < B_OK) {
|
||||
delete device;
|
||||
return B_ERROR;
|
||||
}
|
||||
|
||||
device->fd = -1;
|
||||
device->device_watcher = -1;
|
||||
device->active = false;
|
||||
strcpy(device->path, path);
|
||||
device->device_ref.name = GetShortName(path);
|
||||
device->device_ref.type = B_KEYBOARD_DEVICE;
|
||||
device->device_ref.cookie = device;
|
||||
device->owner = this;
|
||||
|
||||
input_device_ref *devices[2];
|
||||
devices[0] = &device->device_ref;
|
||||
devices[1] = NULL;
|
||||
|
||||
fDevices.AddItem(device);
|
||||
|
||||
InitFromSettings(device);
|
||||
|
||||
return RegisterDevices(devices);
|
||||
}
|
||||
|
||||
|
||||
status_t
|
||||
KeyboardInputDevice::RemoveDevice(const char *path)
|
||||
{
|
||||
int32 i = 0;
|
||||
keyboard_device *device = NULL;
|
||||
while ((device = (keyboard_device *)fDevices.ItemAt(i)) != NULL) {
|
||||
if (!strcmp(device->path, path)) {
|
||||
free(device->device_ref.name);
|
||||
if (device->fd >= 0)
|
||||
close(device->fd);
|
||||
fDevices.RemoveItem(device);
|
||||
delete device;
|
||||
return B_OK;
|
||||
}
|
||||
}
|
||||
|
||||
return B_ENTRY_NOT_FOUND;
|
||||
}
|
||||
|
||||
|
||||
int32
|
||||
KeyboardInputDevice::DeviceWatcher(void *arg)
|
||||
{
|
||||
KeyboardInputDevice *dev = (KeyboardInputDevice *)arg;
|
||||
keyboard_device *dev = (keyboard_device *)arg;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
char *
|
||||
KeyboardInputDevice::GetShortName(const char *longName)
|
||||
{
|
||||
BString string(longName);
|
||||
BString name;
|
||||
|
||||
int32 slash = string.FindLast("/");
|
||||
int32 previousSlash = string.FindLast("/", slash) + 1;
|
||||
string.CopyInto(name, slash, string.Length() - slash);
|
||||
|
||||
int32 deviceIndex = atoi(name.String()) + 1;
|
||||
string.CopyInto(name, previousSlash, slash - previousSlash);
|
||||
name << " Keyboard " << deviceIndex;
|
||||
|
||||
return strdup(name.String());
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
KeyboardInputDevice::RecursiveScan(const char *directory)
|
||||
{
|
||||
BEntry entry;
|
||||
BDirectory dir(directory);
|
||||
while (dir.GetNextEntry(&entry) == B_OK) {
|
||||
BPath path;
|
||||
entry.GetPath(&path);
|
||||
if (entry.IsDirectory())
|
||||
RecursiveScan(path.Path());
|
||||
else
|
||||
AddDevice(path.Path());
|
||||
}
|
||||
}
|
||||
|
@ -28,7 +28,7 @@
|
||||
#define __KEYBOARDINPUTDEVICE_H
|
||||
|
||||
#include <InputServerDevice.h>
|
||||
#include <InterfaceDefs.h>
|
||||
#include <List.h>
|
||||
#include <stdio.h>
|
||||
|
||||
typedef struct {
|
||||
@ -50,16 +50,18 @@ public:
|
||||
uint32 command, BMessage *message);
|
||||
private:
|
||||
status_t HandleMonitor(BMessage *message);
|
||||
status_t InitFromSettings(uint32 opcode = 0);
|
||||
status_t InitFromSettings(void *cookie, uint32 opcode = 0);
|
||||
void RecursiveScan(const char *directory);
|
||||
|
||||
thread_id fThread;
|
||||
int fFd;
|
||||
bool fQuit;
|
||||
FILE *fLogFile;
|
||||
status_t AddDevice(const char *path);
|
||||
status_t RemoveDevice(const char *path);
|
||||
|
||||
static int32 DeviceWatcher(void *arg);
|
||||
static char *GetShortName(const char *longName);
|
||||
|
||||
kb_settings fSettings;
|
||||
BList fDevices;
|
||||
|
||||
static FILE *sLogFile;
|
||||
};
|
||||
|
||||
extern "C" BInputServerDevice *instantiate_input_device();
|
||||
|
Loading…
Reference in New Issue
Block a user