236 lines
6.2 KiB
C++
236 lines
6.2 KiB
C++
|
#define LOG_TAG "KeyLayoutMap"
|
||
|
|
||
|
#include "KeyLayoutMap.h"
|
||
|
#include <sys/types.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <unistd.h>
|
||
|
#include <errno.h>
|
||
|
#include <utils/String8.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <ui/KeycodeLabels.h>
|
||
|
#include <utils/Log.h>
|
||
|
|
||
|
namespace android {
|
||
|
|
||
|
KeyLayoutMap::KeyLayoutMap()
|
||
|
:m_status(NO_INIT),
|
||
|
m_keys()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
KeyLayoutMap::~KeyLayoutMap()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
static String8
|
||
|
next_token(char const** p, int *line)
|
||
|
{
|
||
|
bool begun = false;
|
||
|
const char* begin = *p;
|
||
|
const char* end = *p;
|
||
|
while (true) {
|
||
|
if (*end == '\n') {
|
||
|
(*line)++;
|
||
|
}
|
||
|
switch (*end)
|
||
|
{
|
||
|
case '#':
|
||
|
if (begun) {
|
||
|
*p = end;
|
||
|
return String8(begin, end-begin);
|
||
|
} else {
|
||
|
do {
|
||
|
begin++;
|
||
|
end++;
|
||
|
} while (*begin != '\0' && *begin != '\n');
|
||
|
}
|
||
|
case '\0':
|
||
|
case ' ':
|
||
|
case '\n':
|
||
|
case '\r':
|
||
|
case '\t':
|
||
|
if (begun || (*end == '\0')) {
|
||
|
*p = end;
|
||
|
return String8(begin, end-begin);
|
||
|
} else {
|
||
|
begin++;
|
||
|
end++;
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
end++;
|
||
|
begun = true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static int32_t
|
||
|
token_to_value(const char *literal, const KeycodeLabel *list)
|
||
|
{
|
||
|
while (list->literal) {
|
||
|
if (0 == strcmp(literal, list->literal)) {
|
||
|
return list->value;
|
||
|
}
|
||
|
list++;
|
||
|
}
|
||
|
return list->value;
|
||
|
}
|
||
|
|
||
|
status_t
|
||
|
KeyLayoutMap::load(const char* filename)
|
||
|
{
|
||
|
int fd = open(filename, O_RDONLY);
|
||
|
if (fd < 0) {
|
||
|
LOGE("error opening file=%s err=%s\n", filename, strerror(errno));
|
||
|
m_status = errno;
|
||
|
return errno;
|
||
|
}
|
||
|
|
||
|
off_t len = lseek(fd, 0, SEEK_END);
|
||
|
off_t errlen = lseek(fd, 0, SEEK_SET);
|
||
|
if (len < 0 || errlen < 0) {
|
||
|
close(fd);
|
||
|
LOGE("error seeking file=%s err=%s\n", filename, strerror(errno));
|
||
|
m_status = errno;
|
||
|
return errno;
|
||
|
}
|
||
|
|
||
|
char* buf = (char*)malloc(len+1);
|
||
|
if (read(fd, buf, len) != len) {
|
||
|
LOGE("error reading file=%s err=%s\n", filename, strerror(errno));
|
||
|
m_status = errno != 0 ? errno : ((int)NOT_ENOUGH_DATA);
|
||
|
return errno != 0 ? errno : ((int)NOT_ENOUGH_DATA);
|
||
|
}
|
||
|
errno = 0;
|
||
|
buf[len] = '\0';
|
||
|
|
||
|
int32_t scancode = -1;
|
||
|
int32_t keycode = -1;
|
||
|
uint32_t flags = 0;
|
||
|
uint32_t tmp;
|
||
|
char* end;
|
||
|
status_t err = NO_ERROR;
|
||
|
int line = 1;
|
||
|
char const* p = buf;
|
||
|
enum { BEGIN, SCANCODE, KEYCODE, FLAG } state = BEGIN;
|
||
|
while (true) {
|
||
|
String8 token = next_token(&p, &line);
|
||
|
if (*p == '\0') {
|
||
|
break;
|
||
|
}
|
||
|
switch (state)
|
||
|
{
|
||
|
case BEGIN:
|
||
|
if (token == "key") {
|
||
|
state = SCANCODE;
|
||
|
} else {
|
||
|
LOGE("%s:%d: expected key, got '%s'\n", filename, line,
|
||
|
token.string());
|
||
|
err = BAD_VALUE;
|
||
|
goto done;
|
||
|
}
|
||
|
break;
|
||
|
case SCANCODE:
|
||
|
scancode = strtol(token.string(), &end, 0);
|
||
|
if (*end != '\0') {
|
||
|
LOGE("%s:%d: expected scancode (a number), got '%s'\n",
|
||
|
filename, line, token.string());
|
||
|
goto done;
|
||
|
}
|
||
|
//LOGI("%s:%d: got scancode %d\n", filename, line, scancode );
|
||
|
state = KEYCODE;
|
||
|
break;
|
||
|
case KEYCODE:
|
||
|
keycode = token_to_value(token.string(), KEYCODES);
|
||
|
//LOGI("%s:%d: got keycode %d for %s\n", filename, line, keycode, token.string() );
|
||
|
if (keycode == 0) {
|
||
|
LOGE("%s:%d: expected keycode, got '%s'\n",
|
||
|
filename, line, token.string());
|
||
|
goto done;
|
||
|
}
|
||
|
state = FLAG;
|
||
|
break;
|
||
|
case FLAG:
|
||
|
if (token == "key") {
|
||
|
if (scancode != -1) {
|
||
|
//LOGI("got key decl scancode=%d keycode=%d"
|
||
|
// " flags=0x%08x\n", scancode, keycode, flags);
|
||
|
Key k = { keycode, flags };
|
||
|
m_keys.add(scancode, k);
|
||
|
state = SCANCODE;
|
||
|
scancode = -1;
|
||
|
keycode = -1;
|
||
|
flags = 0;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
tmp = token_to_value(token.string(), FLAGS);
|
||
|
//LOGI("%s:%d: got flags %x for %s\n", filename, line, tmp, token.string() );
|
||
|
if (tmp == 0) {
|
||
|
LOGE("%s:%d: expected flag, got '%s'\n",
|
||
|
filename, line, token.string());
|
||
|
goto done;
|
||
|
}
|
||
|
flags |= tmp;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (state == FLAG && scancode != -1 ) {
|
||
|
//LOGI("got key decl scancode=%d keycode=%d"
|
||
|
// " flags=0x%08x\n", scancode, keycode, flags);
|
||
|
Key k = { keycode, flags };
|
||
|
m_keys.add(scancode, k);
|
||
|
}
|
||
|
|
||
|
done:
|
||
|
free(buf);
|
||
|
close(fd);
|
||
|
|
||
|
m_status = err;
|
||
|
return err;
|
||
|
}
|
||
|
|
||
|
status_t
|
||
|
KeyLayoutMap::map(int32_t scancode, int32_t *keycode, uint32_t *flags) const
|
||
|
{
|
||
|
if (m_status != NO_ERROR) {
|
||
|
return m_status;
|
||
|
}
|
||
|
|
||
|
ssize_t index = m_keys.indexOfKey(scancode);
|
||
|
if (index < 0) {
|
||
|
//LOGW("couldn't map scancode=%d\n", scancode);
|
||
|
return NAME_NOT_FOUND;
|
||
|
}
|
||
|
|
||
|
const Key& k = m_keys.valueAt(index);
|
||
|
|
||
|
*keycode = k.keycode;
|
||
|
*flags = k.flags;
|
||
|
|
||
|
//LOGD("mapped scancode=%d to keycode=%d flags=0x%08x\n", scancode,
|
||
|
// keycode, flags);
|
||
|
|
||
|
return NO_ERROR;
|
||
|
}
|
||
|
|
||
|
status_t
|
||
|
KeyLayoutMap::findScancodes(int32_t keycode, Vector<int32_t>* outScancodes) const
|
||
|
{
|
||
|
if (m_status != NO_ERROR) {
|
||
|
return m_status;
|
||
|
}
|
||
|
|
||
|
const size_t N = m_keys.size();
|
||
|
for (size_t i=0; i<N; i++) {
|
||
|
if (m_keys.valueAt(i).keycode == keycode) {
|
||
|
outScancodes->add(m_keys.keyAt(i));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return NO_ERROR;
|
||
|
}
|
||
|
|
||
|
};
|