blob: 8feb07d20fd6eea5fc23287e30efde4001c052b1 [file] [log] [blame]
/*
* Copyright (C) 2020 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 TLOG_TAG "coverage-client-srv"
#include "coverage.h"
#include <interface/coverage/client.h>
#include <lib/coverage/common/ipc.h>
#include <lib/coverage/common/shm.h>
#include <lib/tipc/tipc_srv.h>
#include <stdbool.h>
#include <stdlib.h>
#include <string.h>
#include <trusty_log.h>
#include <uapi/err.h>
struct chan_ctx {
struct coverage_record* record;
};
static void broadcast_event(struct shm* mailbox, size_t idx, int event) {
int* app_mailbox = (int*)(mailbox->base) + idx;
WRITE_ONCE(*app_mailbox, event);
}
static int handle_open(handle_t chan,
struct coverage_client_req* req,
struct list_node* coverage_record_list,
struct chan_ctx* ctx) {
int rc;
struct coverage_client_resp resp;
struct coverage_record* record;
char uuid_str[UUID_STR_SIZE];
record = find_coverage_record(coverage_record_list, &req->open_args.uuid);
if (!record) {
uuid_to_str(&req->open_args.uuid, uuid_str);
TLOGE("coverage record not found for uuid: %s\n", uuid_str);
return ERR_NOT_FOUND;
}
resp.hdr.cmd = req->hdr.cmd | COVERAGE_CLIENT_CMD_RESP_BIT;
resp.open_args.record_len = record->record_len;
rc = coverage_send(chan, &resp, sizeof(resp), NULL);
if (rc != NO_ERROR) {
TLOGE("failed (%d) to reply to open request\n", rc);
return rc;
}
ctx->record = record;
return NO_ERROR;
}
static int handle_share_record(handle_t chan,
struct coverage_client_req* req,
struct coverage_record* record,
handle_t memref,
struct shm* mailbox) {
int rc;
struct coverage_client_resp resp;
if (memref == INVALID_IPC_HANDLE) {
TLOGE("invalid memref");
return ERR_BAD_LEN;
}
resp.hdr.cmd = req->hdr.cmd | COVERAGE_CLIENT_CMD_RESP_BIT;
rc = coverage_send(chan, &resp, sizeof(resp), NULL);
if (rc != NO_ERROR) {
TLOGE("failed (%d) to reply to share record request\n", rc);
return rc;
}
shm_init(&record->data, memref, NULL, req->share_record_args.shm_len);
broadcast_event(mailbox, record->idx, COVERAGE_MAILBOX_RECORD_READY);
return NO_ERROR;
}
static int on_connect(const struct tipc_port* port,
handle_t chan,
const struct uuid* peer,
void** ctx_p) {
struct chan_ctx* ctx = malloc(sizeof(*ctx));
if (!ctx) {
TLOGE("failed to allocate channel context\n");
return ERR_NO_MEMORY;
}
ctx->record = NULL;
*ctx_p = ctx;
return NO_ERROR;
}
static int on_message(const struct tipc_port* port, handle_t chan, void* _ctx) {
int rc;
handle_t memref;
struct coverage_client_req req;
struct chan_ctx* ctx = (struct chan_ctx*)_ctx;
struct srv_state* state = get_srv_state(port);
rc = coverage_recv(chan, &req, sizeof(req), &memref);
if (rc != NO_ERROR) {
TLOGE("failed (%d) to receive coverage client request\n", rc);
return rc;
}
switch (req.hdr.cmd) {
case COVERAGE_CLIENT_CMD_OPEN:
return handle_open(chan, &req, &state->coverage_record_list, ctx);
case COVERAGE_CLIENT_CMD_SHARE_RECORD:
return handle_share_record(chan, &req, ctx->record, memref,
&state->mailbox);
default:
TLOGE("cmd 0x%x: unknown command\n", req.hdr.cmd);
return ERR_CMD_UNKNOWN;
}
}
static void on_channel_cleanup(void* _ctx) {
struct chan_ctx* ctx = (struct chan_ctx*)_ctx;
free(ctx);
}
int coverage_client_init(struct srv_state* state) {
static struct tipc_port_acl port_acl = {
.flags = IPC_PORT_ALLOW_TA_CONNECT | IPC_PORT_ALLOW_NS_CONNECT,
};
static struct tipc_port port = {
.name = COVERAGE_CLIENT_PORT,
.msg_max_size = MAX(sizeof(struct coverage_client_req),
sizeof(struct coverage_client_resp)),
.msg_queue_len = 1,
.acl = &port_acl,
};
static struct tipc_srv_ops ops = {
.on_connect = on_connect,
.on_message = on_message,
.on_channel_cleanup = on_channel_cleanup,
};
set_srv_state(&port, state);
return tipc_add_service(state->hset, &port, 1, 1, &ops);
}