/*
|
* Copyright (C) 2008 The Android Open Source Project
|
*
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
* you may not use this file except in compliance with the License.
|
* You may obtain a copy of the License at
|
*
|
* http://www.apache.org/licenses/LICENSE-2.0
|
*
|
* Unless required by applicable law or agreed to in writing, software
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
* See the License for the specific language governing permissions and
|
* limitations under the License.
|
*/
|
|
#define LOG_TAG "FrameworkListener"
|
|
#include <errno.h>
|
#include <stdlib.h>
|
#include <string.h>
|
#include <unistd.h>
|
|
#include <log/log.h>
|
#include <sysutils/FrameworkCommand.h>
|
#include <sysutils/FrameworkListener.h>
|
#include <sysutils/SocketClient.h>
|
|
static const int CMD_BUF_SIZE = 4096;
|
|
FrameworkListener::FrameworkListener(const char *socketName, bool withSeq) :
|
SocketListener(socketName, true, withSeq) {
|
init(socketName, withSeq);
|
}
|
|
FrameworkListener::FrameworkListener(const char *socketName) :
|
SocketListener(socketName, true, false) {
|
init(socketName, false);
|
}
|
|
FrameworkListener::FrameworkListener(int sock) :
|
SocketListener(sock, true) {
|
init(nullptr, false);
|
}
|
|
void FrameworkListener::init(const char* /*socketName*/, bool withSeq) {
|
errorRate = 0;
|
mCommandCount = 0;
|
mWithSeq = withSeq;
|
mSkipToNextNullByte = false;
|
}
|
|
bool FrameworkListener::onDataAvailable(SocketClient *c) {
|
char buffer[CMD_BUF_SIZE];
|
int len;
|
|
len = TEMP_FAILURE_RETRY(read(c->getSocket(), buffer, sizeof(buffer)));
|
if (len < 0) {
|
SLOGE("read() failed (%s)", strerror(errno));
|
return false;
|
} else if (!len) {
|
return false;
|
} else if (buffer[len-1] != '\0') {
|
SLOGW("String is not zero-terminated");
|
android_errorWriteLog(0x534e4554, "29831647");
|
c->sendMsg(500, "Command too large for buffer", false);
|
mSkipToNextNullByte = true;
|
return true;
|
}
|
|
int offset = 0;
|
int i;
|
|
for (i = 0; i < len; i++) {
|
if (buffer[i] == '\0') {
|
/* IMPORTANT: dispatchCommand() expects a zero-terminated string */
|
if (mSkipToNextNullByte) {
|
mSkipToNextNullByte = false;
|
} else {
|
dispatchCommand(c, buffer + offset);
|
}
|
offset = i + 1;
|
}
|
}
|
|
mSkipToNextNullByte = false;
|
return true;
|
}
|
|
void FrameworkListener::registerCmd(FrameworkCommand *cmd) {
|
mCommands.push_back(cmd);
|
}
|
|
void FrameworkListener::dispatchCommand(SocketClient *cli, char *data) {
|
int argc = 0;
|
char *argv[FrameworkListener::CMD_ARGS_MAX];
|
char tmp[CMD_BUF_SIZE];
|
char *p = data;
|
char *q = tmp;
|
char *qlimit = tmp + sizeof(tmp) - 1;
|
bool esc = false;
|
bool quote = false;
|
bool haveCmdNum = !mWithSeq;
|
|
memset(argv, 0, sizeof(argv));
|
memset(tmp, 0, sizeof(tmp));
|
while(*p) {
|
if (*p == '\\') {
|
if (esc) {
|
if (q >= qlimit)
|
goto overflow;
|
*q++ = '\\';
|
esc = false;
|
} else
|
esc = true;
|
p++;
|
continue;
|
} else if (esc) {
|
if (*p == '"') {
|
if (q >= qlimit)
|
goto overflow;
|
*q++ = '"';
|
} else if (*p == '\\') {
|
if (q >= qlimit)
|
goto overflow;
|
*q++ = '\\';
|
} else {
|
cli->sendMsg(500, "Unsupported escape sequence", false);
|
goto out;
|
}
|
p++;
|
esc = false;
|
continue;
|
}
|
|
if (*p == '"') {
|
if (quote)
|
quote = false;
|
else
|
quote = true;
|
p++;
|
continue;
|
}
|
|
if (q >= qlimit)
|
goto overflow;
|
*q = *p++;
|
if (!quote && *q == ' ') {
|
*q = '\0';
|
if (!haveCmdNum) {
|
char *endptr;
|
int cmdNum = (int)strtol(tmp, &endptr, 0);
|
if (endptr == nullptr || *endptr != '\0') {
|
cli->sendMsg(500, "Invalid sequence number", false);
|
goto out;
|
}
|
cli->setCmdNum(cmdNum);
|
haveCmdNum = true;
|
} else {
|
if (argc >= CMD_ARGS_MAX)
|
goto overflow;
|
argv[argc++] = strdup(tmp);
|
}
|
memset(tmp, 0, sizeof(tmp));
|
q = tmp;
|
continue;
|
}
|
q++;
|
}
|
|
*q = '\0';
|
if (argc >= CMD_ARGS_MAX)
|
goto overflow;
|
argv[argc++] = strdup(tmp);
|
#if 0
|
for (int k = 0; k < argc; k++) {
|
SLOGD("arg[%d] = '%s'", k, argv[k]);
|
}
|
#endif
|
|
if (quote) {
|
cli->sendMsg(500, "Unclosed quotes error", false);
|
goto out;
|
}
|
|
if (errorRate && (++mCommandCount % errorRate == 0)) {
|
/* ignore this command - let the timeout handler handle it */
|
SLOGE("Faking a timeout");
|
goto out;
|
}
|
|
for (auto* c : mCommands) {
|
if (!strcmp(argv[0], c->getCommand())) {
|
if (c->runCommand(cli, argc, argv)) {
|
SLOGW("Handler '%s' error (%s)", c->getCommand(), strerror(errno));
|
}
|
goto out;
|
}
|
}
|
cli->sendMsg(500, "Command not recognized", false);
|
out:
|
int j;
|
for (j = 0; j < argc; j++)
|
free(argv[j]);
|
return;
|
|
overflow:
|
cli->sendMsg(500, "Command too long", false);
|
goto out;
|
}
|