blob: d1a4b3e0bcbdbd37539e574538c5e82c90b32604 [file] [log] [blame]
/*
* Copyright (C) 2019 The Android Open Source Project
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use, copy,
* modify, merge, publish, distribute, sublicense, and/or sell copies
* of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
#include <stddef.h>
#include <test-runner-arch.h>
#include <trusty/sysdeps.h>
#include <utils.h>
#include <virtio-console.h>
#include <virtio.h>
enum msg_type {
MSG_LOG,
MSG_EXIT,
};
struct msg_log {
unsigned char size;
char data[];
} __attribute__((__packed__));
struct msg_exit {
char status;
} __attribute__((__packed__));
struct msg {
unsigned char type;
union {
struct msg_log log;
struct msg_exit exit;
};
} __attribute__((__packed__));
static const char* testruner_msg_devname = "testrunner0";
static struct virtq input;
static struct virtq_raw input_raw;
static struct virtq output;
static struct virtq_raw output_raw;
#define TESTRUNNER0_BUF_SIZE 64
static char msg_buf[TESTRUNNER0_BUF_SIZE];
static int testrunner_msg_scan(struct virtio_console* console) {
for (size_t i = 0; i < MAX_PORTS; i++) {
if (!trusty_strcmp(testruner_msg_devname, console->ports[i].name)) {
if (!console->ports[i].host_connected) {
return -1;
}
return i;
}
}
return -1;
}
int init_log(struct virtio_console* console) {
int port_id = testrunner_msg_scan(console);
if (port_id < 0) {
return -1;
}
vq_init(&input, &input_raw, console->vio, true);
vq_init(&output, &output_raw, console->vio, false);
virtio_console_connect_port(console, port_id, &input, &output);
return 0;
}
size_t host_get_cmdline(char* buf, size_t size) {
uint32_t len;
if (input.raw) {
len = recv_vq(&input, buf, size);
/*
* Make sure message is always null terminated, message buffer
* exceed maxium size would be dropped.
*/
len = MIN(size - 1, len);
buf[len] = '\0';
return len;
} else {
return 0;
}
}
void log_buf(const char* buf, size_t size) {
size_t remain = size;
size_t offset = 0;
size_t copied = 0;
struct msg* log_msg = (struct msg*)&msg_buf;
log_msg->type = MSG_LOG;
while (remain) {
copied = MIN((TESTRUNNER0_BUF_SIZE - sizeof(struct msg)), remain);
log_msg->log.size = copied;
trusty_memcpy(log_msg->log.data, buf + offset, copied);
/*
* Always send all message buffers to host each time, it makes
* host side easy to handle messages. With this logic, each
* message received in host side contains a message header and
* a message body only.
*/
send_vq(&output, msg_buf, TESTRUNNER0_BUF_SIZE);
remain -= copied;
offset += copied;
}
}
void host_exit(uint32_t code) {
struct msg* exit_msg = (struct msg*)&msg_buf;
exit_msg->type = MSG_EXIT;
exit_msg->exit.status = code;
send_vq(&output, msg_buf, TESTRUNNER0_BUF_SIZE);
}
void log_msg(const char* msg) {
log_buf(msg, trusty_strlen(msg));
}
void abort_msg(const char* msg) {
log_msg(msg);
host_exit(2);
}