/* * 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 #include #include #include #include #include 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); }